123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 |
- # python-cc1101 - Python Library to Transmit RF Signals via C1101 Transceivers
- #
- # Copyright (C) 2020 Fabian Peter Hammerle <fabian@hammerle.me>
- #
- # This program is free software: you can redistribute it and/or modify
- # it under the terms of the GNU General Public License as published by
- # the Free Software Foundation, either version 3 of the License, or
- # any later version.
- #
- # This program is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- # GNU General Public License for more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program. If not, see <https://www.gnu.org/licenses/>.
- import unittest.mock
- import pytest
- import cc1101
- from cc1101.options import PacketLengthMode, SyncMode
- # pylint: disable=protected-access
- @pytest.mark.parametrize(
- ("xfer_return_value", "sync_word"),
- [([64, 211, 145], b"\xd3\x91"), ([64, 0, 0], b"\0\0")],
- )
- def test_get_sync_word(transceiver, xfer_return_value, sync_word):
- transceiver._spi.xfer.return_value = xfer_return_value
- assert transceiver.get_sync_word() == sync_word
- transceiver._spi.xfer.assert_called_once_with([0x04 | 0xC0, 0, 0])
- def test_set_sync_word(transceiver):
- transceiver._spi.xfer.return_value = [15, 15, 15]
- transceiver.set_sync_word(b"\x12\x34")
- transceiver._spi.xfer.assert_called_once_with([0x04 | 0x40, 0x12, 0x34])
- @pytest.mark.parametrize("sync_word", [b"", b"\0", "\x12\x34\x56"])
- def test_set_sync_word_invalid_length(transceiver, sync_word):
- with pytest.raises(ValueError, match=r"\bexpected two bytes\b"):
- transceiver.set_sync_word(sync_word)
- _FREQUENCY_CONTROL_WORD_HERTZ_PARAMS = [
- ([0x10, 0xA7, 0x62], 433000000),
- ([0x10, 0xAB, 0x85], 433420000),
- ([0x10, 0xB1, 0x3B], 434000000),
- ([0x21, 0x62, 0x76], 868000000),
- ]
- @pytest.mark.parametrize(
- ("control_word", "hertz"), _FREQUENCY_CONTROL_WORD_HERTZ_PARAMS
- )
- def test__frequency_control_word_to_hertz(control_word, hertz):
- assert cc1101.CC1101._frequency_control_word_to_hertz(
- control_word
- ) == pytest.approx(hertz, abs=200)
- @pytest.mark.parametrize(
- ("control_word", "hertz"), _FREQUENCY_CONTROL_WORD_HERTZ_PARAMS
- )
- def test__hertz_to_frequency_control_word(control_word, hertz):
- assert cc1101.CC1101._hertz_to_frequency_control_word(hertz) == control_word
- _FILTER_BANDWIDTH_MANTISSA_EXPONENT_REAL_PARAMS = [
- # > The default values give 203 kHz channel filter bandwidth,
- # > assuming a 26.0 MHz crystal.
- (0, 2, 203e3),
- # "Table 26: Channel Filter Bandwidths [kHz] (assuming a 26 MHz crystal)"
- (0, 0, 812e3),
- (0, 1, 406e3),
- (0, 2, 203e3),
- (1, 0, 650e3),
- (1, 1, 325e3),
- (3, 0, 464e3),
- (3, 1, 232e3),
- (3, 2, 116e3),
- (3, 3, 58e3),
- ]
- @pytest.mark.parametrize(
- ("mantissa", "exponent", "real"), _FILTER_BANDWIDTH_MANTISSA_EXPONENT_REAL_PARAMS
- )
- def test__filter_bandwidth_floating_point_to_real(mantissa, exponent, real):
- assert cc1101.CC1101._filter_bandwidth_floating_point_to_real(
- mantissa=mantissa, exponent=exponent
- ) == pytest.approx(real, rel=1e-3)
- @pytest.mark.parametrize(
- ("mdmcfg4", "real"),
- [
- (0b10001100, 203e3),
- (0b10001010, 203e3),
- (0b10001110, 203e3),
- (0b11111100, 58e3),
- (0b01011100, 325e3),
- ],
- )
- def test__get_filter_bandwidth_hertz(transceiver, mdmcfg4, real):
- transceiver._spi.xfer.return_value = [15, mdmcfg4]
- assert transceiver._get_filter_bandwidth_hertz() == pytest.approx(real, rel=1e-3)
- transceiver._spi.xfer.assert_called_once_with([0x10 | 0x80, 0])
- @pytest.mark.parametrize(
- ("mdmcfg4_before", "mdmcfg4_after", "exponent", "mantissa"),
- [
- (0b00001010, 0b10111010, 0b10, 0b11),
- (0b00001100, 0b01001100, 0b01, 0b00),
- (0b00001100, 0b10111100, 0b10, 0b11),
- (0b00001100, 0b11011100, 0b11, 0b01),
- (0b01011100, 0b11011100, 0b11, 0b01),
- (0b11111100, 0b11011100, 0b11, 0b01),
- ],
- )
- def test__set_filter_bandwidth(
- transceiver, mdmcfg4_before, mdmcfg4_after, exponent, mantissa
- ):
- transceiver._spi.xfer.return_value = [15, 15]
- with unittest.mock.patch.object(
- transceiver, "_read_single_byte", return_value=mdmcfg4_before
- ):
- transceiver._set_filter_bandwidth(mantissa=mantissa, exponent=exponent)
- transceiver._spi.xfer.assert_called_once_with([0x10 | 0x40, mdmcfg4_after])
- @pytest.mark.parametrize(
- ("mdmcfg4", "symbol_rate_exponent"), [(0b1001100, 12), (0b10011001, 9)]
- )
- def test__get_symbol_rate_exponent(transceiver, mdmcfg4, symbol_rate_exponent):
- transceiver._spi.xfer.return_value = [15, mdmcfg4]
- assert transceiver._get_symbol_rate_exponent() == symbol_rate_exponent
- transceiver._spi.xfer.assert_called_once_with([0x10 | 0x80, 0])
- @pytest.mark.parametrize(
- ("mdmcfg3", "symbol_rate_mantissa"), [(0b00100010, 34), (0b10101010, 170)]
- )
- def test__get_symbol_rate_mantissa(transceiver, mdmcfg3, symbol_rate_mantissa):
- transceiver._spi.xfer.return_value = [15, mdmcfg3]
- assert transceiver._get_symbol_rate_mantissa() == symbol_rate_mantissa
- transceiver._spi.xfer.assert_called_once_with([0x11 | 0x80, 0])
- _SYMBOL_RATE_MANTISSA_EXPONENT_REAL_PARAMS = [
- # > The default values give a data rate of 115.051 kBaud
- # > (closest setting to 115.2 kBaud), assuming a 26.0 MHz crystal.
- (34, 12, 115051),
- (34, 12 + 1, 115051 * 2),
- (34, 12 - 1, 115051 / 2),
- ]
- @pytest.mark.parametrize(
- ("mantissa", "exponent", "real"), _SYMBOL_RATE_MANTISSA_EXPONENT_REAL_PARAMS
- )
- def test__symbol_rate_floating_point_to_real(mantissa, exponent, real):
- assert cc1101.CC1101._symbol_rate_floating_point_to_real(
- mantissa=mantissa, exponent=exponent
- ) == pytest.approx(real, rel=1e-5)
- @pytest.mark.parametrize(
- ("mantissa", "exponent", "real"), _SYMBOL_RATE_MANTISSA_EXPONENT_REAL_PARAMS
- )
- def test__symbol_rate_real_to_floating_point(mantissa, exponent, real):
- assert cc1101.CC1101._symbol_rate_real_to_floating_point(real) == (
- mantissa,
- exponent,
- )
- @pytest.mark.parametrize(
- ("mdmcfg2", "sync_mode"),
- [
- (0b00000000, SyncMode.NO_PREAMBLE_AND_SYNC_WORD),
- (0b00000001, SyncMode.TRANSMIT_16_MATCH_15_BITS),
- (0b00000010, SyncMode.TRANSMIT_16_MATCH_16_BITS),
- (0b00000011, SyncMode.TRANSMIT_32_MATCH_30_BITS),
- (0b00000110, SyncMode.TRANSMIT_16_MATCH_16_BITS),
- (0b00000111, SyncMode.TRANSMIT_32_MATCH_30_BITS),
- (0b00001100, SyncMode.NO_PREAMBLE_AND_SYNC_WORD),
- (0b01101011, SyncMode.TRANSMIT_32_MATCH_30_BITS),
- (0b01101111, SyncMode.TRANSMIT_32_MATCH_30_BITS),
- ],
- )
- def test_get_sync_mode(transceiver, mdmcfg2, sync_mode):
- transceiver._spi.xfer.return_value = [15, mdmcfg2]
- assert transceiver.get_sync_mode() == sync_mode
- transceiver._spi.xfer.assert_called_once_with([0x12 | 0x80, 0])
- @pytest.mark.parametrize(
- ("mdmcfg2_before", "mdmcfg2_after", "sync_mode", "threshold_enabled"),
- [
- (0b00000010, 0b00000000, SyncMode.NO_PREAMBLE_AND_SYNC_WORD, None),
- (0b00000010, 0b00000001, SyncMode.TRANSMIT_16_MATCH_15_BITS, None),
- (0b00000010, 0b00000010, SyncMode.TRANSMIT_16_MATCH_16_BITS, None),
- (0b00000010, 0b00000011, SyncMode.TRANSMIT_32_MATCH_30_BITS, None),
- (0b01101110, 0b01101111, SyncMode.TRANSMIT_32_MATCH_30_BITS, None),
- (0b00000010, 0b00000110, SyncMode.TRANSMIT_16_MATCH_16_BITS, True),
- (0b00000010, 0b00000111, SyncMode.TRANSMIT_32_MATCH_30_BITS, True),
- (0b01101110, 0b01101111, SyncMode.TRANSMIT_32_MATCH_30_BITS, True),
- (0b00000010, 0b00000010, SyncMode.TRANSMIT_16_MATCH_16_BITS, False),
- (0b00000010, 0b00000011, SyncMode.TRANSMIT_32_MATCH_30_BITS, False),
- (0b01101110, 0b01101011, SyncMode.TRANSMIT_32_MATCH_30_BITS, False),
- ],
- )
- def test_set_sync_mode(
- transceiver, mdmcfg2_before, mdmcfg2_after, sync_mode, threshold_enabled
- ):
- transceiver._spi.xfer.return_value = [15, 15]
- with unittest.mock.patch.object(
- transceiver, "_read_single_byte", return_value=mdmcfg2_before
- ):
- transceiver.set_sync_mode(
- sync_mode, _carrier_sense_threshold_enabled=threshold_enabled
- )
- transceiver._spi.xfer.assert_called_once_with([0x12 | 0x40, mdmcfg2_after])
- @pytest.mark.parametrize(
- ("mdmcfg1", "length"),
- [
- (0b00000010, 2),
- (0b00010010, 3),
- (0b00100010, 4),
- (0b00110010, 6),
- (0b01000010, 8),
- (0b01010010, 12),
- (0b01100010, 16),
- (0b01110010, 24),
- ],
- )
- def test_get_preamble_length_bytes(transceiver, mdmcfg1, length):
- transceiver._spi.xfer.return_value = [0, mdmcfg1]
- assert transceiver.get_preamble_length_bytes() == length
- transceiver._spi.xfer.assert_called_once_with([0x13 | 0x80, 0])
- @pytest.mark.parametrize(
- ("mdmcfg1_before", "mdmcfg1_after", "length"),
- [
- (0b00000010, 0b00000010, 2),
- (0b00000010, 0b00010010, 3),
- (0b00000010, 0b00100010, 4),
- (0b00000010, 0b00110010, 6),
- (0b00000010, 0b01000010, 8),
- (0b00000010, 0b01010010, 12),
- (0b00000010, 0b01100010, 16),
- (0b00000010, 0b01110010, 24),
- (0b01010010, 0b01100010, 16),
- (0b01110010, 0b00000010, 2),
- (0b01110010, 0b01000010, 8),
- (0b11011010, 0b11101010, 16),
- (0b11110111, 0b11000111, 8),
- (0b11111110, 0b10001110, 2),
- ],
- )
- def test_set_preamble_length_bytes(transceiver, mdmcfg1_before, mdmcfg1_after, length):
- transceiver._spi.xfer.return_value = [15, 15]
- with unittest.mock.patch.object(
- transceiver, "_read_single_byte", return_value=mdmcfg1_before
- ):
- transceiver.set_preamble_length_bytes(length)
- transceiver._spi.xfer.assert_called_once_with([0x13 | 0x40, mdmcfg1_after])
- @pytest.mark.parametrize(
- "length", [-21, 0, 1, 5, 7, 9, 10, 11, 13, 14, 15, 17, 20, 23, 25, 32, 42]
- )
- def test_set_preamble_length_bytes_invalid(transceiver, length):
- with pytest.raises(ValueError, match=r"\bpreamble length\b"):
- transceiver.set_preamble_length_bytes(length)
- def test_get_packet_length_bytes(transceiver):
- xfer_mock = transceiver._spi.xfer
- xfer_mock.return_value = [0, 8]
- assert transceiver.get_packet_length_bytes() == 8
- xfer_mock.assert_called_once_with([0x06 | 0x80, 0])
- @pytest.mark.parametrize("packet_length", [21])
- def test_set_packet_length_bytes(transceiver, packet_length):
- xfer_mock = transceiver._spi.xfer
- xfer_mock.return_value = [15, 15]
- transceiver.set_packet_length_bytes(packet_length)
- xfer_mock.assert_called_once_with([0x06 | 0x40, packet_length])
- @pytest.mark.parametrize("packet_length", [-21, 0, 256, 1024])
- def test_set_packet_length_bytes_fail(transceiver, packet_length):
- with pytest.raises(Exception):
- transceiver.set_packet_length_bytes(packet_length)
- transceiver._spi.xfer.assert_not_called()
- @pytest.mark.parametrize(
- ("pktctrl0_before", "pktctrl0_after"),
- (
- # unchanged
- (0b00000000, 0b00000000),
- (0b00010000, 0b00010000),
- (0b00010001, 0b00010001),
- (0b01000000, 0b01000000),
- (0b01000010, 0b01000010),
- (0b01110000, 0b01110000),
- (0b01110010, 0b01110010),
- # disabled
- (0b00010100, 0b00010000),
- (0b01000100, 0b01000000),
- (0b01000110, 0b01000010),
- (0b01110110, 0b01110010),
- ),
- )
- def test_disable_checksum(transceiver, pktctrl0_before, pktctrl0_after):
- xfer_mock = transceiver._spi.xfer
- xfer_mock.return_value = [15, 15]
- with unittest.mock.patch.object(
- transceiver, "_read_single_byte", return_value=pktctrl0_before
- ):
- transceiver.disable_checksum()
- xfer_mock.assert_called_once_with([0x08 | 0x40, pktctrl0_after])
- @pytest.mark.parametrize(
- ("pktctrl0", "expected_mode"),
- (
- (0b00000000, PacketLengthMode.FIXED),
- (0b00000001, PacketLengthMode.VARIABLE),
- (0b01000100, PacketLengthMode.FIXED),
- (0b01000101, PacketLengthMode.VARIABLE),
- ),
- )
- def test_get_packet_length_mode(transceiver, pktctrl0, expected_mode):
- xfer_mock = transceiver._spi.xfer
- xfer_mock.return_value = [0, pktctrl0]
- assert transceiver.get_packet_length_mode() == expected_mode
- xfer_mock.assert_called_once_with([0x08 | 0x80, 0])
- @pytest.mark.parametrize(
- ("pktctrl0_before", "pktctrl0_after", "mode"),
- (
- (0b00000000, 0b00000000, PacketLengthMode.FIXED),
- (0b00000001, 0b00000000, PacketLengthMode.FIXED),
- (0b00000001, 0b00000001, PacketLengthMode.VARIABLE),
- (0b00000010, 0b00000000, PacketLengthMode.FIXED),
- (0b00000010, 0b00000001, PacketLengthMode.VARIABLE),
- (0b01000100, 0b01000100, PacketLengthMode.FIXED),
- (0b01000100, 0b01000101, PacketLengthMode.VARIABLE),
- (0b01000101, 0b01000100, PacketLengthMode.FIXED),
- (0b01000101, 0b01000101, PacketLengthMode.VARIABLE),
- ),
- )
- def test_set_packet_length_mode(transceiver, pktctrl0_before, pktctrl0_after, mode):
- xfer_mock = transceiver._spi.xfer
- xfer_mock.return_value = [15, 15]
- with unittest.mock.patch.object(
- transceiver, "_read_single_byte", return_value=pktctrl0_before
- ):
- transceiver.set_packet_length_mode(mode)
- xfer_mock.assert_called_once_with([0x08 | 0x40, pktctrl0_after])
|