/** @file A two-dimensional bit buffer consisting of bytes. Copyright (C) 2015 Tommy Vestermark 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 2 of the License, or (at your option) any later version. */ #ifndef INCLUDE_BITBUFFER_H_ #define INCLUDE_BITBUFFER_H_ #include // NOTE: Wireless mbus protocol needs at least ((256+16*2+3)*12)/8 => 437 bytes // which fits even if RTL_433_REDUCE_STACK_USE is defined because of row spilling #ifdef RTL_433_REDUCE_STACK_USE #define BITBUF_COLS 40 // Number of bytes in a column #define BITBUF_ROWS 25 #else #define BITBUF_COLS 128 // Number of bytes in a column #define BITBUF_ROWS 50 #endif #define BITBUF_MAX_ROW_BITS (BITBUF_ROWS * BITBUF_COLS * 8) // Maximum number of bits per row, max UINT16_MAX #define BITBUF_MAX_PRINT_BITS 50 // Maximum number of bits to print (in addition to hex values) typedef uint8_t bitrow_t[BITBUF_COLS]; typedef bitrow_t bitarray_t[BITBUF_ROWS]; /// Bit buffer. typedef struct bitbuffer { uint16_t num_rows; ///< Number of active rows uint16_t free_row; ///< Index of next free row uint16_t bits_per_row[BITBUF_ROWS]; ///< Number of active bits per row uint16_t syncs_before_row[BITBUF_ROWS]; ///< Number of sync pulses before row bitarray_t bb; ///< The actual bits buffer } bitbuffer_t; /// Clear the content of the bitbuffer. void bitbuffer_clear(bitbuffer_t *bits); /// Add a single bit at the end of the bitbuffer (MSB first). void bitbuffer_add_bit(bitbuffer_t *bits, int bit); /// Add a new row to the bitbuffer. void bitbuffer_add_row(bitbuffer_t *bits); /// Increment sync counter, add new row if not empty. void bitbuffer_add_sync(bitbuffer_t *bits); /// Extract (potentially unaligned) bytes from the bit buffer. Len is bits. void bitbuffer_extract_bytes(bitbuffer_t *bitbuffer, unsigned row, unsigned pos, uint8_t *out, unsigned len); /// Invert all bits in the bitbuffer (do not invert the empty bits). void bitbuffer_invert(bitbuffer_t *bits); /// Non-Return-to-Zero Space (NRZI) decode the bitbuffer. /// "One" is represented by no change in level, "Zero" is represented by change in level. void bitbuffer_nrzs_decode(bitbuffer_t *bits); /// Non-Return-to-Zero Mark (NRZI) decode the bitbuffer. /// "One" is represented by change in level, "Zero" is represented by no change in level. void bitbuffer_nrzm_decode(bitbuffer_t *bits); /// Print the content of the bitbuffer. /// @deprecated For debug only, use decoder_log_bitbuffer otherwise void bitbuffer_print(const bitbuffer_t *bits); /// Debug the content of the bitbuffer. /// @deprecated For debug only, use decoder_log_bitbuffer otherwise void bitbuffer_debug(const bitbuffer_t *bits); /// Print the content of a bit row (byte buffer). /// @deprecated For debug only, use decoder_log_bitrow otherwise void bitrow_print(uint8_t const *bitrow, unsigned bit_len); /// Debug the content of a bit row (byte buffer). /// @deprecated For debug only, use decoder_log_bitrow otherwise void bitrow_debug(uint8_t const *bitrow, unsigned bit_len); /// Print the content of a bit row (byte buffer) to a string buffer. /// /// Write at most @p size - 1 characters, /// the output is always null-terminated, unless size is 0. /// /// @param bitrow the row of bytes to print /// @param bit_len the number of bits in @p bitrow to print /// @param str an output string buffer of sufficient size /// @param size the size of @p str /// /// @return the number of characters printed (not including the trailing `\0`). int bitrow_snprint(uint8_t const *bitrow, unsigned bit_len, char *str, unsigned size); /// Parse a string into a bitbuffer. /// /// The (optionally "0x" prefixed) hex code is processed into a bitbuffer_t. /// Each row is optionally prefixed with a length enclosed in braces "{}" or /// separated with a slash "/" character. Whitespace is ignored. void bitbuffer_parse(bitbuffer_t *bits, const char *code); /// Search the specified row of the bitbuffer, starting from bit 'start', for /// the pattern provided. /// /// The pattern starts in the high bit. For example if searching for 011011 /// the byte pointed to by 'pattern' would be 0xAC. (011011xx). /// /// @return the location of the first match, or the end of the row if no match is found. unsigned bitbuffer_search(bitbuffer_t *bitbuffer, unsigned row, unsigned start, const uint8_t *pattern, unsigned pattern_bits_len); /// Manchester decoding from one bitbuffer into another, starting at the /// specified row and start bit. /// /// Decode at most 'max' data bits (i.e. 2*max) bits from the input buffer). /// Manchester per IEEE 802.3 conventions, i.e. high-low is a 0 bit, low-high is a 1 bit. /// /// @return the bit position in the input row /// (i.e. returns start + 2*outbuf->bits_per_row[0]). unsigned bitbuffer_manchester_decode(bitbuffer_t *inbuf, unsigned row, unsigned start, bitbuffer_t *outbuf, unsigned max); /// Differential Manchester decoding from one bitbuffer into another, starting at the /// specified row and start bit. /// /// Decode at most 'max' data bits (i.e. 2*max) bits from the input buffer). /// /// @return the bit position in the input row /// (i.e. returns start + 2*outbuf->bits_per_row[0]). unsigned bitbuffer_differential_manchester_decode(bitbuffer_t *inbuf, unsigned row, unsigned start, bitbuffer_t *outbuf, unsigned max); /// Compares two given rows of a bitbuffer. /// /// If @p max_bits is greater than 0 then only up that many bits are compared. int bitbuffer_compare_rows(bitbuffer_t *bits, unsigned row_a, unsigned row_b, unsigned max_bits); /// Count the number of repeats of row at index @p row. /// /// If @p max_bits is greater than 0 then only up that many bits are compared. /// The returned count will include the given row and will be at least 1. unsigned bitbuffer_count_repeats(bitbuffer_t *bits, unsigned row, unsigned max_bits); /// Find a row repeated at least @p min_repeats times and with at least @p min_bits bits length, /// all bits in the repeats need to match. /// @return the row index or -1. int bitbuffer_find_repeated_row(bitbuffer_t *bits, unsigned min_repeats, unsigned min_bits); /// Find a row repeated at least @p min_repeats times and with at least @p min_bits bits length, /// a prefix of at most @p min_bits bits will be compared. /// @return the row index or -1. int bitbuffer_find_repeated_prefix(bitbuffer_t *bits, unsigned min_repeats, unsigned min_bits); /// Return a single bit from a bitrow at bit_idx position. static inline uint8_t bitrow_get_bit(uint8_t const *bitrow, unsigned bit_idx) { return bitrow[bit_idx >> 3] >> (7 - (bit_idx & 7)) & 1; } /// Return a single byte from a bitrow at bit_idx position (which may be unaligned). static inline uint8_t bitrow_get_byte(uint8_t const *bitrow, unsigned bit_idx) { return (uint8_t)((bitrow[(bit_idx >> 3)] << (bit_idx & 7)) | (bitrow[(bit_idx >> 3) + 1] >> (8 - (bit_idx & 7)))); } #endif /* INCLUDE_BITBUFFER_H_ */