protocol_ds1996.c 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. #include "protocol_ds1996.h"
  2. #include <core/core_defines.h>
  3. #include <toolbox/pretty_format.h>
  4. #include "dallas_common.h"
  5. #define DS1996_FAMILY_CODE 0x0CU
  6. #define DS1996_FAMILY_NAME "DS1996"
  7. #define DS1996_SRAM_DATA_SIZE 8192U
  8. #define DS1996_SRAM_PAGE_SIZE 32U
  9. #define DS1996_COPY_SCRATCH_TIMEOUT_US 100U
  10. #define DS1996_DATA_BYTE_COUNT 4U
  11. #define DS1996_SRAM_DATA_KEY "Sram Data"
  12. typedef struct {
  13. OneWireSlave* bus;
  14. DallasCommonCommandState command_state;
  15. } DS1996ProtocolState;
  16. typedef struct {
  17. DallasCommonRomData rom_data;
  18. uint8_t sram_data[DS1996_SRAM_DATA_SIZE];
  19. DS1996ProtocolState state;
  20. } DS1996ProtocolData;
  21. static bool dallas_ds1996_read(OneWireHost*, void*);
  22. static bool dallas_ds1996_write_copy(OneWireHost*, iButtonProtocolData*);
  23. static void dallas_ds1996_emulate(OneWireSlave*, iButtonProtocolData*);
  24. static bool dallas_ds1996_load(FlipperFormat*, uint32_t, iButtonProtocolData*);
  25. static bool dallas_ds1996_save(FlipperFormat*, const iButtonProtocolData*);
  26. static void dallas_ds1996_render_data(FuriString*, const iButtonProtocolData*);
  27. static void dallas_ds1996_render_brief_data(FuriString*, const iButtonProtocolData*);
  28. static void dallas_ds1996_render_error(FuriString*, const iButtonProtocolData*);
  29. static bool dallas_ds1996_is_data_valid(const iButtonProtocolData*);
  30. static void dallas_ds1996_get_editable_data(iButtonEditableData*, iButtonProtocolData*);
  31. static void dallas_ds1996_apply_edits(iButtonProtocolData*);
  32. const iButtonProtocolDallasBase ibutton_protocol_ds1996 = {
  33. .family_code = DS1996_FAMILY_CODE,
  34. .features = iButtonProtocolFeatureExtData | iButtonProtocolFeatureWriteCopy,
  35. .data_size = sizeof(DS1996ProtocolData),
  36. .manufacturer = DALLAS_COMMON_MANUFACTURER_NAME,
  37. .name = DS1996_FAMILY_NAME,
  38. .read = dallas_ds1996_read,
  39. .write_blank = NULL, /* Data too big for known blanks */
  40. .write_copy = dallas_ds1996_write_copy,
  41. .emulate = dallas_ds1996_emulate,
  42. .save = dallas_ds1996_save,
  43. .load = dallas_ds1996_load,
  44. .render_data = dallas_ds1996_render_data,
  45. .render_brief_data = dallas_ds1996_render_brief_data,
  46. .render_error = dallas_ds1996_render_error,
  47. .is_valid = dallas_ds1996_is_data_valid,
  48. .get_editable_data = dallas_ds1996_get_editable_data,
  49. .apply_edits = dallas_ds1996_apply_edits,
  50. };
  51. bool dallas_ds1996_read(OneWireHost* host, iButtonProtocolData* protocol_data) {
  52. DS1996ProtocolData* data = protocol_data;
  53. return onewire_host_reset(host) && dallas_common_read_rom(host, &data->rom_data) &&
  54. dallas_common_read_mem(host, 0, data->sram_data, DS1996_SRAM_DATA_SIZE);
  55. }
  56. bool dallas_ds1996_write_copy(OneWireHost* host, iButtonProtocolData* protocol_data) {
  57. DS1996ProtocolData* data = protocol_data;
  58. return dallas_common_write_mem(
  59. host,
  60. DS1996_COPY_SCRATCH_TIMEOUT_US,
  61. DS1996_SRAM_PAGE_SIZE,
  62. data->sram_data,
  63. DS1996_SRAM_DATA_SIZE);
  64. }
  65. static void dallas_ds1996_reset_callback(void* context) {
  66. furi_assert(context);
  67. DS1996ProtocolData* data = context;
  68. data->state.command_state = DallasCommonCommandStateIdle;
  69. }
  70. static bool dallas_ds1996_command_callback(uint8_t command, void* context) {
  71. furi_assert(context);
  72. DS1996ProtocolData* data = context;
  73. OneWireSlave* bus = data->state.bus;
  74. switch(command) {
  75. case DALLAS_COMMON_CMD_SEARCH_ROM:
  76. if(data->state.command_state == DallasCommonCommandStateIdle) {
  77. data->state.command_state = DallasCommonCommandStateRomCmd;
  78. return dallas_common_emulate_search_rom(bus, &data->rom_data);
  79. } else if(data->state.command_state == DallasCommonCommandStateRomCmd) {
  80. data->state.command_state = DallasCommonCommandStateMemCmd;
  81. dallas_common_emulate_read_mem(bus, data->sram_data, DS1996_SRAM_DATA_SIZE);
  82. return false;
  83. } else {
  84. return false;
  85. }
  86. case DALLAS_COMMON_CMD_READ_ROM:
  87. if(data->state.command_state == DallasCommonCommandStateIdle) {
  88. data->state.command_state = DallasCommonCommandStateRomCmd;
  89. return dallas_common_emulate_read_rom(bus, &data->rom_data);
  90. } else {
  91. return false;
  92. }
  93. case DALLAS_COMMON_CMD_SKIP_ROM:
  94. if(data->state.command_state == DallasCommonCommandStateIdle) {
  95. data->state.command_state = DallasCommonCommandStateRomCmd;
  96. return true;
  97. } else {
  98. return false;
  99. }
  100. case DALLAS_COMMON_CMD_OVERDRIVE_SKIP_ROM:
  101. case DALLAS_COMMON_CMD_OVERDRIVE_MATCH_ROM:
  102. /* TODO: Overdrive mode support */
  103. default:
  104. return false;
  105. }
  106. }
  107. void dallas_ds1996_emulate(OneWireSlave* bus, iButtonProtocolData* protocol_data) {
  108. DS1996ProtocolData* data = protocol_data;
  109. data->state.bus = bus;
  110. onewire_slave_set_reset_callback(bus, dallas_ds1996_reset_callback, protocol_data);
  111. onewire_slave_set_command_callback(bus, dallas_ds1996_command_callback, protocol_data);
  112. }
  113. bool dallas_ds1996_load(
  114. FlipperFormat* ff,
  115. uint32_t format_version,
  116. iButtonProtocolData* protocol_data) {
  117. DS1996ProtocolData* data = protocol_data;
  118. bool success = false;
  119. do {
  120. if(format_version < 2) break;
  121. if(!dallas_common_load_rom_data(ff, format_version, &data->rom_data)) break;
  122. if(!flipper_format_read_hex(
  123. ff, DS1996_SRAM_DATA_KEY, data->sram_data, DS1996_SRAM_DATA_SIZE))
  124. break;
  125. success = true;
  126. } while(false);
  127. return success;
  128. }
  129. bool dallas_ds1996_save(FlipperFormat* ff, const iButtonProtocolData* protocol_data) {
  130. const DS1996ProtocolData* data = protocol_data;
  131. bool success = false;
  132. do {
  133. if(!dallas_common_save_rom_data(ff, &data->rom_data)) break;
  134. if(!flipper_format_write_hex(
  135. ff, DS1996_SRAM_DATA_KEY, data->sram_data, DS1996_SRAM_DATA_SIZE))
  136. break;
  137. success = true;
  138. } while(false);
  139. return success;
  140. }
  141. void dallas_ds1996_render_data(FuriString* result, const iButtonProtocolData* protocol_data) {
  142. const DS1996ProtocolData* data = protocol_data;
  143. pretty_format_bytes_hex_canonical(
  144. result,
  145. DS1996_DATA_BYTE_COUNT,
  146. PRETTY_FORMAT_FONT_MONOSPACE,
  147. data->sram_data,
  148. DS1996_SRAM_DATA_SIZE);
  149. }
  150. void dallas_ds1996_render_brief_data(FuriString* result, const iButtonProtocolData* protocol_data) {
  151. const DS1996ProtocolData* data = protocol_data;
  152. dallas_common_render_brief_data(
  153. result, &data->rom_data, data->sram_data, DS1996_SRAM_DATA_SIZE);
  154. }
  155. void dallas_ds1996_render_error(FuriString* result, const iButtonProtocolData* protocol_data) {
  156. const DS1996ProtocolData* data = protocol_data;
  157. if(!dallas_common_is_valid_crc(&data->rom_data)) {
  158. dallas_common_render_crc_error(result, &data->rom_data);
  159. }
  160. }
  161. bool dallas_ds1996_is_data_valid(const iButtonProtocolData* protocol_data) {
  162. const DS1996ProtocolData* data = protocol_data;
  163. return dallas_common_is_valid_crc(&data->rom_data);
  164. }
  165. void dallas_ds1996_get_editable_data(
  166. iButtonEditableData* editable_data,
  167. iButtonProtocolData* protocol_data) {
  168. DS1996ProtocolData* data = protocol_data;
  169. editable_data->ptr = data->rom_data.bytes;
  170. editable_data->size = sizeof(DallasCommonRomData);
  171. }
  172. void dallas_ds1996_apply_edits(iButtonProtocolData* protocol_data) {
  173. DS1996ProtocolData* data = protocol_data;
  174. dallas_common_apply_edits(&data->rom_data, DS1996_FAMILY_CODE);
  175. }