DSPWriter.cpp 5.4 KB

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