/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2017 - ROLI Ltd. JUCE is an open source library subject to commercial or open-source licensing. The code included in this file is provided under the terms of the ISC license http://www.isc.org/downloads/software-support-policy/isc-license. Permission To use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted provided that the above copyright notice and this permission notice appear in all copies. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE DISCLAIMED. ============================================================================== */ namespace littlefoot { //============================================================================== /** This class manages the synchronisation of a remote block of heap memory used by a littlefoot program running on a block. Data in the block can be changed by calling setByte, setBytes, setBits etc, and these changes will be flushed to the device when sendChanges is called. @tags{Blocks} */ template struct LittleFootRemoteHeap { LittleFootRemoteHeap (uint32 blockSizeToUse) noexcept : blockSize (blockSizeToUse) { resetDeviceStateToUnknown(); } void clear() noexcept { zeromem (targetData, sizeof (targetData)); invalidateData(); } void resetDeviceStateToUnknown() { invalidateData(); messagesSent.clear(); resetDataRangeToUnknown (0, ImplementationClass::maxBlockSize); } void resetDataRangeToUnknown (size_t offset, size_t size) noexcept { auto* latestState = getLatestExpectedDataState(); for (size_t i = 0; i < size; ++i) latestState[offset + i] = unknownByte; } void setByte (size_t offset, uint8 value) noexcept { if (offset < blockSize) { if (targetData[offset] != value) { targetData[offset] = value; needsSyncing = true; if (programStateKnown && offset < programSize) programStateKnown = false; } } else { jassertfalse; } } void setBytes (size_t offset, const uint8* data, size_t num) noexcept { for (size_t i = 0; i < num; ++i) setByte (offset + i, data[i]); } void setBits (uint32 startBit, uint32 numBits, uint32 value) noexcept { if (startBit + numBits > 8 * blockSize) { jassertfalse; return; } if (readLittleEndianBitsInBuffer (targetData, startBit, numBits) != value) { writeLittleEndianBitsInBuffer (targetData, startBit, numBits, value); invalidateData(); } } uint8 getByte (size_t offset) noexcept { if (offset < blockSize) return targetData [offset]; jassertfalse; return 0; } void invalidateData() noexcept { needsSyncing = true; programStateKnown = false; } bool isFullySynced() const noexcept { return ! needsSyncing; } static bool isAllZero (const uint8* data, size_t size) noexcept { for (size_t i = 0; i < size; ++i) if (data[i] != 0) return false; return true; } void sendChanges (ImplementationClass& bi, bool forceSend) { if ((needsSyncing && messagesSent.isEmpty()) || forceSend) { for (int maxChanges = 30; --maxChanges >= 0;) { if (isAllZero (targetData, blockSize)) break; uint16 data[ImplementationClass::maxBlockSize]; auto* latestState = getLatestExpectedDataState(); for (uint32 i = 0; i < blockSize; ++i) data[i] = latestState[i]; uint32 packetIndex = messagesSent.isEmpty() ? lastPacketIndexReceived : messagesSent.getLast()->packetIndex; packetIndex = (packetIndex + 1) & ImplementationClass::maxPacketCounter; if (! Diff (data, targetData, blockSize).createChangeMessage (bi, data, messagesSent, packetIndex)) break; dumpStatus(); } } for (auto* m : messagesSent) { if (m->dispatchTime >= Time::getCurrentTime() - RelativeTime::milliseconds (250)) break; m->dispatchTime = Time::getCurrentTime(); bi.sendMessageToDevice (m->packet); //DBG ("Sending packet " << (int) m->packetIndex << " - " << m->packet.size() << " bytes, device " << bi.getDeviceIndex()); if (getTotalSizeOfMessagesSent() > 200) break; } } void handleACKFromDevice (ImplementationClass& bi, uint32 packetIndex) noexcept { //DBG ("ACK " << (int) packetIndex << " device " << (int) bi.getDeviceIndex()); if (lastPacketIndexReceived != packetIndex) { lastPacketIndexReceived = packetIndex; for (int i = messagesSent.size(); --i >= 0;) { auto& m = *messagesSent.getUnchecked(i); if (m.packetIndex == packetIndex) { for (uint32 j = 0; j < blockSize; ++j) deviceState[j] = m.resultDataState[j]; messagesSent.removeRange (0, i + 1); dumpStatus(); sendChanges (bi, false); if (messagesSent.isEmpty()) needsSyncing = false; return; } } resetDeviceStateToUnknown(); } } bool isProgramLoaded() noexcept { if (! programStateKnown) { programStateKnown = true; uint8 deviceMemory[ImplementationClass::maxBlockSize]; for (size_t i = 0; i < blockSize; ++i) deviceMemory[i] = (uint8) deviceState[i]; littlefoot::Program prog (deviceMemory, (uint32) blockSize); programLoaded = prog.checksumMatches(); programSize = prog.getProgramSize(); } return programLoaded; } const size_t blockSize; static constexpr uint16 unknownByte = 0x100; private: uint16 deviceState[ImplementationClass::maxBlockSize] = { 0 }; uint8 targetData[ImplementationClass::maxBlockSize] = { 0 }; uint32 programSize = 0; bool needsSyncing = true, programStateKnown = true, programLoaded = false; uint16* getLatestExpectedDataState() noexcept { return messagesSent.isEmpty() ? deviceState : messagesSent.getLast()->resultDataState; } struct ChangeMessage { typename ImplementationClass::PacketBuilder packet; Time dispatchTime; uint32 packetIndex; uint16 resultDataState[ImplementationClass::maxBlockSize]; }; OwnedArray messagesSent; uint32 lastPacketIndexReceived = 0; int getTotalSizeOfMessagesSent() const noexcept { int total = 0; for (auto* m : messagesSent) if (m->dispatchTime != Time()) total += m->packet.size(); return total; } void dumpStatus() { #if DUMP_LITTLEFOOT_HEAP_STATUS int differences = 0; constexpr int diffLen = 50; char areas[diffLen + 1] = { 0 }; for (int i = 0; i < diffLen; ++i) areas[i] = '.'; for (int i = 0; i < (int) blockSize; ++i) { if (targetData[i] != deviceState[i]) { ++differences; areas[i * diffLen / (int) blockSize] = 'X'; } } double proportionOK = ((int) blockSize - differences) / (double) blockSize; juce::ignoreUnused (proportionOK); DBG ("Heap: " << areas << " " << String (roundToInt (100 * proportionOK)) << "% " << (isProgramLoaded() ? "Ready" : "Loading")); #endif } struct Diff { Diff (uint16* current, const uint8* target, size_t blockSizeToUse) : newData (target), blockSize (blockSizeToUse) { ranges.ensureStorageAllocated ((int) blockSize); for (int i = 0; i < (int) blockSize; ++i) ranges.add ({ i, 1, newData[i] == current[i], false }); coalesceUniformRegions(); coalesceSequences(); trim(); } bool createChangeMessage (const ImplementationClass& bi, const uint16* currentState, OwnedArray& messagesCreated, uint32 nextPacketIndex) { if (ranges.isEmpty()) return false; auto deviceIndex = bi.getDeviceIndex(); if (deviceIndex < 0) return false; auto& message = *messagesCreated.add (new ChangeMessage()); message.packetIndex = nextPacketIndex; for (uint32 i = 0; i < blockSize; ++i) message.resultDataState[i] = currentState[i]; auto& p = message.packet; p.writePacketSysexHeaderBytes ((uint8) deviceIndex); p.beginDataChanges (nextPacketIndex); uint8 lastValue = 0; bool packetOverflow = false; for (auto& r : ranges) { if (r.isSkipped) { packetOverflow = ! p.skipBytes (r.length); } else if (r.isMixed) { jassert (r.length > 1); packetOverflow = ! p.setMultipleBytes (newData + r.index, r.length); if (! packetOverflow) lastValue = newData[r.index + r.length - 1]; } else { auto value = newData[r.index]; packetOverflow = ! p.setMultipleBytes (value, lastValue, r.length); if (! packetOverflow) lastValue = value; } if (packetOverflow) break; if (! r.isSkipped) for (int i = r.index; i < r.index + r.length; ++i) message.resultDataState[i] = newData[i]; } p.endDataChanges (! packetOverflow); p.writePacketSysexFooter(); return packetOverflow; } private: struct ByteSequence { int index, length; bool isSkipped, isMixed; }; const uint8* const newData; const size_t blockSize; Array ranges; void coalesceUniformRegions() { for (int i = ranges.size(); --i > 0;) { auto& r1 = ranges.getReference (i - 1); auto r2 = ranges.getReference (i); if (r1.isSkipped == r2.isSkipped && (r1.isSkipped || newData[r1.index] == newData[r2.index])) { r1.length += r2.length; ranges.remove (i); i = jmin (ranges.size() - 1, i + 1); } } } void coalesceSequences() { for (int i = ranges.size(); --i > 0;) { auto& r1 = ranges.getReference (i - 1); auto r2 = ranges.getReference (i); if (! (r1.isSkipped || r2.isSkipped) && (r1.isMixed || r1.length == 1) && (r2.isMixed || r2.length == 1)) { if (r1.length + r2.length < 32) { r1.length += r2.length; r1.isMixed = true; ranges.remove (i); i = jmin (ranges.size() - 1, i + 1); } } } } void trim() { while (ranges.size() > 0 && ranges.getLast().isSkipped) ranges.removeLast(); } }; }; }