DSPWriter.cpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. #include <Arduino.h>
  2. #include "DSPWriter.h"
  3. #include <Wire.h>
  4. #include "DataConversion.h"
  5. static uint8_t s_safeload_count = 0;
  6. static uint8_t s_coreRegCache[2] = {0x00, 0x00};
  7. static constexpr uint8_t IST_MASK = 0x3C;
  8. DSPWriter::DSPWriter() {}
  9. DSPWriter::~DSPWriter() {}
  10. void DSPWriter::downloadProgram()
  11. {
  12. /*
  13. DSPWriter::writeRegisterBlock(REG_COREREGISTER_IC_1_ADDR, REG_COREREGISTER_IC_1_BYTE, R0_COREREGISTER_IC_1_Default, CORE_REGISTER_R0_REGSIZE);
  14. DSPWriter::writeRegisterBlock(PROGRAM_ADDR_IC_1, PROGRAM_SIZE_IC_1, Program_Data_IC_1, PROGRAM_REGSIZE);
  15. DSPWriter::writeRegisterBlock(PARAM_ADDR_IC_1, PARAM_SIZE_IC_1, Param_Data_IC_1, PARAMETER_REGSIZE);
  16. DSPWriter::writeRegisterBlock(REG_COREREGISTER_IC_1_ADDR, R3_HWCONFIGURATION_IC_1_SIZE, R3_HWCONFIGURATION_IC_1_Default, HARDWARE_CONF_REGSIZE);
  17. DSPWriter::writeRegisterBlock(REG_COREREGISTER_IC_1_ADDR, REG_COREREGISTER_IC_1_BYTE, R4_COREREGISTER_IC_1_Default, CORE_REGISTER_R4_REGSIZE);*/
  18. }
  19. bool DSPWriter::writeRegisterBlock(uint16_t subAddress, int dataLength, const uint8_t* pdata, uint8_t registerSize)
  20. {
  21. uint16_t bytesSent = 0;
  22. while (bytesSent < dataLength)
  23. {
  24. uint8_t MSByte = subAddress >> 8;
  25. uint8_t LSByte = (uint8_t)(subAddress & 0xFF);
  26. Wire.beginTransmission(DSP_I2C_ADDRESS);
  27. Wire.write(MSByte);
  28. Wire.write(LSByte);
  29. uint8_t chunk = registerSize;
  30. if ((bytesSent + chunk) > dataLength) {
  31. chunk = dataLength - bytesSent;
  32. }
  33. for (uint8_t i = 0; i < chunk; i++) {
  34. Wire.write(pdata[bytesSent++]);
  35. }
  36. uint8_t err = Wire.endTransmission();
  37. if (err != 0) {
  38. // 1 = data too long, 2 = NACK addr, 3 = NACK data, 4 = other
  39. Serial.printf("I2C NACK addr=0x%04X err=%u\n", subAddress, err);
  40. return false;
  41. }
  42. subAddress++;
  43. delay(0);
  44. }
  45. return true;
  46. }
  47. void DSPWriter::writeRegister(uint16_t memoryAddress, uint8_t length, const uint8_t* data)
  48. {
  49. if (memoryAddress == dspRegister::CoreRegister && length >= 2) {
  50. s_coreRegCache[0] = data[0];
  51. s_coreRegCache[1] = data[1];
  52. }
  53. uint8_t LSByte = (uint8_t)memoryAddress & 0xFF;
  54. uint8_t MSByte = memoryAddress >> 8;
  55. Wire.beginTransmission(DSP_I2C_ADDRESS);
  56. Wire.write(MSByte);
  57. Wire.write(LSByte);
  58. for (uint8_t i = 0; i < length; i++)
  59. Wire.write(data[i]);
  60. Wire.endTransmission();
  61. }
  62. // ---------------------------------------------------------------
  63. // Safeload counter — static so it survives across DSPWriter
  64. // instances, but exposed via resetSafeload() so the TCP bridge
  65. // can flush and reset cleanly on session start and disconnect.
  66. // ---------------------------------------------------------------
  67. void DSPWriter::resetSafeload()
  68. {
  69. // If the session ended mid-safeload, the DSP still has its own internal
  70. // safeload counter pointing at those stale slots. Triggering IST here
  71. // commits whatever is pending to parameter RAM so the DSP counter resets
  72. // to zero before the next session writes new safeload data.
  73. if (s_safeload_count > 0) {
  74. uint8_t ist[2] = { s_coreRegCache[0], (uint8_t)(s_coreRegCache[1] | IST_MASK) };
  75. DSPWriter::writeRegister(dspRegister::CoreRegister, sizeof(ist), ist);
  76. }
  77. s_safeload_count = 0;
  78. }
  79. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, uint8_t* data, bool finished)
  80. {
  81. uint8_t addr[2]; // Address array
  82. addr[0] = (memoryAddress >> 8) & 0xFF;
  83. addr[1] = memoryAddress & 0xFF;
  84. // Place the 16-bit memory address into the next safeload address slot
  85. DSPWriter::writeRegister(dspRegister::SafeloadAddress0 + s_safeload_count, sizeof(addr), addr);
  86. // Q: Why is the safeload register five bytes long for four-byte parameters?
  87. // A: Safeload registers also support five-byte slew RAM writes. For normal
  88. // parameter RAM writes the first byte is always 0x00.
  89. DSPWriter::writeRegister(dspRegister::SafeloadData0 + s_safeload_count, 5, data);
  90. s_safeload_count++;
  91. if (finished == true || s_safeload_count >= 5) // Max 5 safeload slots
  92. {
  93. uint8_t ist[2] = { s_coreRegCache[0], (uint8_t)(s_coreRegCache[1] | IST_MASK) };
  94. DSPWriter::writeRegister(dspRegister::CoreRegister, sizeof(ist), ist);
  95. s_safeload_count = 0;
  96. }
  97. }
  98. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, int32_t data, bool finished)
  99. {
  100. uint8_t dataArray[5];
  101. DataConversion::intToFixed(data, dataArray);
  102. safeload_writeRegister(memoryAddress, dataArray, finished);
  103. }
  104. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, float data, bool finished)
  105. {
  106. uint8_t dataArray[5];
  107. DataConversion::floatToFixed(data, dataArray);
  108. safeload_writeRegister(memoryAddress, dataArray, finished);
  109. }
  110. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, int16_t data, bool finished) { safeload_writeRegister(memoryAddress, (int32_t)data, finished); }
  111. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, uint32_t data, bool finished) { safeload_writeRegister(memoryAddress, (int32_t)data, finished); }
  112. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, uint16_t data, bool finished) { safeload_writeRegister(memoryAddress, (int32_t)data, finished); }
  113. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, uint8_t data, bool finished) { safeload_writeRegister(memoryAddress, (int32_t)data, finished); }
  114. void DSPWriter::safeload_writeRegister(uint16_t memoryAddress, double data, bool finished) { safeload_writeRegister(memoryAddress, (float)data, finished); }