protocol_ds1996.c 7.4 KB

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