|
22 | 22 | #include <lib/support/CHIPMem.h>
|
23 | 23 | #include <tracing/esp32_diagnostic_trace/Diagnostics.h>
|
24 | 24 |
|
| 25 | +#define TLV_CLOSING_BYTES 4 |
| 26 | + |
25 | 27 | namespace chip {
|
26 | 28 | namespace Tracing {
|
27 | 29 | namespace Diagnostics {
|
28 |
| -class DiagnosticStorageImpl : public DiagnosticStorageInterface |
| 30 | +class CircularDiagnosticBuffer : public chip::TLV::TLVCircularBuffer, public DiagnosticStorageInterface |
29 | 31 | {
|
30 | 32 | public:
|
31 |
| - static DiagnosticStorageImpl & GetInstance(uint8_t * buffer = nullptr, size_t bufferSize = 0); |
| 33 | + // Singleton instance getter |
| 34 | + static CircularDiagnosticBuffer & GetInstance() |
| 35 | + { |
| 36 | + static CircularDiagnosticBuffer instance; |
| 37 | + return instance; |
| 38 | + } |
32 | 39 |
|
33 |
| - DiagnosticStorageImpl(const DiagnosticStorageImpl &) = delete; |
34 |
| - DiagnosticStorageImpl & operator=(const DiagnosticStorageImpl &) = delete; |
| 40 | + // Delete copy constructor and assignment operator to ensure singleton |
| 41 | + CircularDiagnosticBuffer(const CircularDiagnosticBuffer &) = delete; |
| 42 | + CircularDiagnosticBuffer & operator=(const CircularDiagnosticBuffer &) = delete; |
35 | 43 |
|
36 |
| - CHIP_ERROR Store(DiagnosticEntry & diagnostic) override; |
| 44 | + void Init(uint8_t * buffer, size_t bufferLength) { chip::TLV::TLVCircularBuffer::Init(buffer, bufferLength); } |
37 | 45 |
|
38 |
| - CHIP_ERROR Retrieve(MutableByteSpan & payload) override; |
| 46 | + CHIP_ERROR Store(DiagnosticEntry & entry) override |
| 47 | + { |
| 48 | + chip::TLV::CircularTLVWriter writer; |
| 49 | + writer.Init(*this); |
39 | 50 |
|
40 |
| - bool IsEmptyBuffer(); |
| 51 | + CHIP_ERROR err = entry.Encode(writer); |
| 52 | + if (err != CHIP_NO_ERROR) |
| 53 | + { |
| 54 | + ChipLogError(DeviceLayer, "Failed to write entry: %s", chip::ErrorStr(err)); |
| 55 | + } |
| 56 | + return err; |
| 57 | + } |
41 | 58 |
|
42 |
| - uint32_t GetDataSize(); |
| 59 | + CHIP_ERROR Retrieve(MutableByteSpan & span) override |
| 60 | + { |
| 61 | + CHIP_ERROR err = CHIP_NO_ERROR; |
| 62 | + chip::TLV::TLVReader reader; |
| 63 | + reader.Init(*this); |
43 | 64 |
|
44 |
| -private: |
45 |
| - DiagnosticStorageImpl(uint8_t * buffer, size_t bufferSize); |
46 |
| - DiagnosticStorageImpl(); |
47 |
| - ~DiagnosticStorageImpl(); |
| 65 | + chip::TLV::TLVWriter writer; |
| 66 | + writer.Init(span.data(), span.size()); |
| 67 | + |
| 68 | + chip::TLV::TLVType outWriterContainer; |
| 69 | + err = writer.StartContainer(chip::TLV::AnonymousTag(), chip::TLV::kTLVType_List, outWriterContainer); |
| 70 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, ChipLogError(DeviceLayer, "Failed to start container")); |
| 71 | + |
| 72 | + while (CHIP_NO_ERROR == reader.Next()) |
| 73 | + { |
| 74 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, |
| 75 | + ChipLogError(DeviceLayer, "Failed to read next TLV element: %s", ErrorStr(err))); |
| 76 | + |
| 77 | + if (reader.GetType() == chip::TLV::kTLVType_Structure && reader.GetTag() == chip::TLV::AnonymousTag()) |
| 78 | + { |
| 79 | + chip::TLV::TLVType outerReaderContainer; |
| 80 | + err = reader.EnterContainer(outerReaderContainer); |
| 81 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, |
| 82 | + ChipLogError(DeviceLayer, "Failed to enter outer TLV container: %s", ErrorStr(err))); |
| 83 | + |
| 84 | + err = reader.Next(); |
| 85 | + VerifyOrReturnError( |
| 86 | + err == CHIP_NO_ERROR, err, |
| 87 | + ChipLogError(DeviceLayer, "Failed to read next TLV element in outer container: %s", ErrorStr(err))); |
48 | 88 |
|
49 |
| - chip::TLV::TLVCircularBuffer mEndUserCircularBuffer; |
| 89 | + if ((reader.GetType() == chip::TLV::kTLVType_Structure) && |
| 90 | + (reader.GetTag() == chip::TLV::ContextTag(DIAGNOSTICS_TAG::METRIC) || |
| 91 | + reader.GetTag() == chip::TLV::ContextTag(DIAGNOSTICS_TAG::TRACE) || |
| 92 | + reader.GetTag() == chip::TLV::ContextTag(DIAGNOSTICS_TAG::COUNTER))) |
| 93 | + { |
| 94 | + if ((reader.GetLengthRead() - writer.GetLengthWritten()) < ((writer.GetRemainingFreeLength() + TLV_CLOSING_BYTES))) |
| 95 | + { |
| 96 | + err = writer.CopyElement(reader); |
| 97 | + if (err == CHIP_ERROR_BUFFER_TOO_SMALL) |
| 98 | + { |
| 99 | + ChipLogProgress(DeviceLayer, "Buffer too small to occupy current element"); |
| 100 | + break; |
| 101 | + } |
| 102 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, ChipLogError(DeviceLayer, "Failed to copy TLV element")); |
| 103 | + } |
| 104 | + else |
| 105 | + { |
| 106 | + ChipLogProgress(DeviceLayer, "Buffer too small to occupy current TLV"); |
| 107 | + break; |
| 108 | + } |
| 109 | + } |
| 110 | + else |
| 111 | + { |
| 112 | + ChipLogError(DeviceLayer, "Unexpected TLV element in outer container"); |
| 113 | + reader.ExitContainer(outerReaderContainer); |
| 114 | + return CHIP_ERROR_WRONG_TLV_TYPE; |
| 115 | + } |
| 116 | + err = reader.ExitContainer(outerReaderContainer); |
| 117 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, |
| 118 | + ChipLogError(DeviceLayer, "Failed to exit outer TLV container: %s", ErrorStr(err))); |
| 119 | + } |
| 120 | + else |
| 121 | + { |
| 122 | + ChipLogError(DeviceLayer, "Unexpected TLV element type or tag in outer container"); |
| 123 | + } |
| 124 | + } |
| 125 | + |
| 126 | + err = writer.EndContainer(outWriterContainer); |
| 127 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, ChipLogError(DeviceLayer, "Failed to close outer container")); |
| 128 | + err = writer.Finalize(); |
| 129 | + VerifyOrReturnError(err == CHIP_NO_ERROR, err, ChipLogError(DeviceLayer, "Failed to finalize TLV writing")); |
| 130 | + span.reduce_size(writer.GetLengthWritten()); |
| 131 | + ChipLogProgress(DeviceLayer, "---------------Total written bytes successfully : %ld----------------\n", |
| 132 | + writer.GetLengthWritten()); |
| 133 | + ChipLogProgress(DeviceLayer, "Retrieval successful"); |
| 134 | + return CHIP_NO_ERROR; |
| 135 | + } |
| 136 | + |
| 137 | + bool IsEmptyBuffer() override { return DataLength() == 0; } |
| 138 | + |
| 139 | + uint32_t GetDataSize() override { return DataLength(); } |
| 140 | + |
| 141 | +private: |
| 142 | + CircularDiagnosticBuffer() : chip::TLV::TLVCircularBuffer(nullptr, 0) {} |
| 143 | + ~CircularDiagnosticBuffer() override = default; |
50 | 144 | };
|
| 145 | + |
51 | 146 | } // namespace Diagnostics
|
52 | 147 | } // namespace Tracing
|
53 | 148 | } // namespace chip
|
0 commit comments