DSPWriter.cpp 5.4 KB

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