Device interface library for multiple platforms including Mbed.

Dependents:   DeepCover Embedded Security in IoT MaximInterface MAXREFDES155#

Maxim Interface is a library framework focused on providing flexible and expressive hardware interfaces. Both communication interfaces such as I2C and 1-Wire and device interfaces such as DS18B20 are supported. Modern C++ concepts are used extensively while keeping compatibility with C++98/C++03 and requiring no external dependencies. The embedded-friendly design does not depend on exceptions or RTTI.

The full version of the project is hosted on GitLab: https://gitlab.com/iabenz/MaximInterface

Committer:
IanBenzMaxim
Date:
Mon Jul 22 11:44:07 2019 -0500
Revision:
7:9cd16581b578
Child:
8:5ea891c7d1a1
Updated to version 1.9.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
IanBenzMaxim 7:9cd16581b578 1 /*******************************************************************************
IanBenzMaxim 7:9cd16581b578 2 * Copyright (C) 2018 Maxim Integrated Products, Inc., All Rights Reserved.
IanBenzMaxim 7:9cd16581b578 3 *
IanBenzMaxim 7:9cd16581b578 4 * Permission is hereby granted, free of charge, to any person obtaining a
IanBenzMaxim 7:9cd16581b578 5 * copy of this software and associated documentation files (the "Software"),
IanBenzMaxim 7:9cd16581b578 6 * to deal in the Software without restriction, including without limitation
IanBenzMaxim 7:9cd16581b578 7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
IanBenzMaxim 7:9cd16581b578 8 * and/or sell copies of the Software, and to permit persons to whom the
IanBenzMaxim 7:9cd16581b578 9 * Software is furnished to do so, subject to the following conditions:
IanBenzMaxim 7:9cd16581b578 10 *
IanBenzMaxim 7:9cd16581b578 11 * The above copyright notice and this permission notice shall be included
IanBenzMaxim 7:9cd16581b578 12 * in all copies or substantial portions of the Software.
IanBenzMaxim 7:9cd16581b578 13 *
IanBenzMaxim 7:9cd16581b578 14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
IanBenzMaxim 7:9cd16581b578 15 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
IanBenzMaxim 7:9cd16581b578 16 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
IanBenzMaxim 7:9cd16581b578 17 * IN NO EVENT SHALL MAXIM INTEGRATED BE LIABLE FOR ANY CLAIM, DAMAGES
IanBenzMaxim 7:9cd16581b578 18 * OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
IanBenzMaxim 7:9cd16581b578 19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
IanBenzMaxim 7:9cd16581b578 20 * OTHER DEALINGS IN THE SOFTWARE.
IanBenzMaxim 7:9cd16581b578 21 *
IanBenzMaxim 7:9cd16581b578 22 * Except as contained in this notice, the name of Maxim Integrated
IanBenzMaxim 7:9cd16581b578 23 * Products, Inc. shall not be used except as stated in the Maxim Integrated
IanBenzMaxim 7:9cd16581b578 24 * Products, Inc. Branding Policy.
IanBenzMaxim 7:9cd16581b578 25 *
IanBenzMaxim 7:9cd16581b578 26 * The mere transfer of this software does not imply any licenses
IanBenzMaxim 7:9cd16581b578 27 * of trade secrets, proprietary technology, copyrights, patents,
IanBenzMaxim 7:9cd16581b578 28 * trademarks, maskwork rights, or any other form of intellectual
IanBenzMaxim 7:9cd16581b578 29 * property whatsoever. Maxim Integrated Products, Inc. retains all
IanBenzMaxim 7:9cd16581b578 30 * ownership rights.
IanBenzMaxim 7:9cd16581b578 31 *******************************************************************************/
IanBenzMaxim 7:9cd16581b578 32
IanBenzMaxim 7:9cd16581b578 33 #include <stddef.h>
IanBenzMaxim 7:9cd16581b578 34 #include <algorithm>
IanBenzMaxim 7:9cd16581b578 35 #include <MaximInterfaceCore/Error.hpp>
IanBenzMaxim 7:9cd16581b578 36 #include "DS28E39.hpp"
IanBenzMaxim 7:9cd16581b578 37
IanBenzMaxim 7:9cd16581b578 38 namespace MaximInterfaceDevices {
IanBenzMaxim 7:9cd16581b578 39
IanBenzMaxim 7:9cd16581b578 40 using namespace Core;
IanBenzMaxim 7:9cd16581b578 41 using std::copy;
IanBenzMaxim 7:9cd16581b578 42
IanBenzMaxim 7:9cd16581b578 43 static const int readMemoryTimeMs = 30;
IanBenzMaxim 7:9cd16581b578 44 static const int writeMemoryTimeMs = 65;
IanBenzMaxim 7:9cd16581b578 45 static const int writeStateTimeMs = 15;
IanBenzMaxim 7:9cd16581b578 46 static const int generateEccKeyPairTimeMs = 200;
IanBenzMaxim 7:9cd16581b578 47 static const int generateEcdsaSignatureTimeMs = 130;
IanBenzMaxim 7:9cd16581b578 48 static const int trngOnDemandCheckTimeMs = 20;
IanBenzMaxim 7:9cd16581b578 49 static const int trngGenerationTimeMs = 10;
IanBenzMaxim 7:9cd16581b578 50 static const int verifyEcdsaSignatureTimeMs = 180;
IanBenzMaxim 7:9cd16581b578 51
IanBenzMaxim 7:9cd16581b578 52 const int DS28E39::decrementCounterPage;
IanBenzMaxim 7:9cd16581b578 53 const int DS28E39::authorityPublicKeyXPage;
IanBenzMaxim 7:9cd16581b578 54 const int DS28E39::authorityPublicKeyYPage;
IanBenzMaxim 7:9cd16581b578 55 const int DS28E39::writePublicKeyXPage;
IanBenzMaxim 7:9cd16581b578 56 const int DS28E39::writePublicKeyYPage;
IanBenzMaxim 7:9cd16581b578 57 const int DS28E39::memoryPages;
IanBenzMaxim 7:9cd16581b578 58
IanBenzMaxim 7:9cd16581b578 59 error_code DS28E39::writeMemory(int pageNum, Page::const_span page) {
IanBenzMaxim 7:9cd16581b578 60 if (pageNum < 0 || pageNum >= memoryPages) {
IanBenzMaxim 7:9cd16581b578 61 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 62 }
IanBenzMaxim 7:9cd16581b578 63
IanBenzMaxim 7:9cd16581b578 64 uint_least8_t request[2 + Page::size];
IanBenzMaxim 7:9cd16581b578 65 request[0] = 0x96;
IanBenzMaxim 7:9cd16581b578 66 request[1] = pageNum;
IanBenzMaxim 7:9cd16581b578 67 copy(page.begin(), page.end(), request + 2);
IanBenzMaxim 7:9cd16581b578 68 return runCommand(request, writeMemoryTimeMs);
IanBenzMaxim 7:9cd16581b578 69 }
IanBenzMaxim 7:9cd16581b578 70
IanBenzMaxim 7:9cd16581b578 71 error_code DS28E39::readMemory(int pageNum, Page::span page) {
IanBenzMaxim 7:9cd16581b578 72 if (pageNum < 0 || pageNum >= memoryPages) {
IanBenzMaxim 7:9cd16581b578 73 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 74 }
IanBenzMaxim 7:9cd16581b578 75
IanBenzMaxim 7:9cd16581b578 76 uint_least8_t buffer[1 + Page::size];
IanBenzMaxim 7:9cd16581b578 77 buffer[0] = 0x44;
IanBenzMaxim 7:9cd16581b578 78 buffer[1] = pageNum;
IanBenzMaxim 7:9cd16581b578 79 span<uint_least8_t> response(buffer);
IanBenzMaxim 7:9cd16581b578 80 const error_code result =
IanBenzMaxim 7:9cd16581b578 81 runCommand(make_span(buffer, 2), readMemoryTimeMs, response);
IanBenzMaxim 7:9cd16581b578 82 if (!result) {
IanBenzMaxim 7:9cd16581b578 83 copy(response.begin(), response.end(), page.begin());
IanBenzMaxim 7:9cd16581b578 84 }
IanBenzMaxim 7:9cd16581b578 85 return result;
IanBenzMaxim 7:9cd16581b578 86 }
IanBenzMaxim 7:9cd16581b578 87
IanBenzMaxim 7:9cd16581b578 88 error_code DS28E39::readStatus(bool entropyHealthTest, Status & status) {
IanBenzMaxim 7:9cd16581b578 89 int delay = readMemoryTimeMs;
IanBenzMaxim 7:9cd16581b578 90 if (entropyHealthTest) {
IanBenzMaxim 7:9cd16581b578 91 delay += trngOnDemandCheckTimeMs;
IanBenzMaxim 7:9cd16581b578 92 }
IanBenzMaxim 7:9cd16581b578 93 uint_least8_t buffer[Status::PageProtectionList::csize + ManId::size +
IanBenzMaxim 7:9cd16581b578 94 Status::RomVersion::csize + 2];
IanBenzMaxim 7:9cd16581b578 95 buffer[0] = 0xAA;
IanBenzMaxim 7:9cd16581b578 96 buffer[1] = entropyHealthTest ? 0x01 : 0x00;
IanBenzMaxim 7:9cd16581b578 97 span<uint_least8_t> response(buffer);
IanBenzMaxim 7:9cd16581b578 98 error_code result = runCommand(make_span(buffer, 2), delay, response);
IanBenzMaxim 7:9cd16581b578 99 if (!result) {
IanBenzMaxim 7:9cd16581b578 100 span<uint_least8_t>::const_iterator responseIt = response.begin();
IanBenzMaxim 7:9cd16581b578 101 for (Status::PageProtectionList::iterator it =
IanBenzMaxim 7:9cd16581b578 102 status.pageProtection.begin();
IanBenzMaxim 7:9cd16581b578 103 it != status.pageProtection.end(); ++it) {
IanBenzMaxim 7:9cd16581b578 104 *it = *responseIt;
IanBenzMaxim 7:9cd16581b578 105 ++responseIt;
IanBenzMaxim 7:9cd16581b578 106 }
IanBenzMaxim 7:9cd16581b578 107 span<uint_least8_t>::const_iterator responseItEnd =
IanBenzMaxim 7:9cd16581b578 108 responseIt + status.manId.size();
IanBenzMaxim 7:9cd16581b578 109 copy(responseIt, responseItEnd, status.manId.begin());
IanBenzMaxim 7:9cd16581b578 110 responseIt = responseItEnd;
IanBenzMaxim 7:9cd16581b578 111 responseItEnd = responseIt + status.romVersion.size();
IanBenzMaxim 7:9cd16581b578 112 copy(responseIt, responseItEnd, status.romVersion.begin());
IanBenzMaxim 7:9cd16581b578 113 responseIt = responseItEnd;
IanBenzMaxim 7:9cd16581b578 114 switch (*responseIt) {
IanBenzMaxim 7:9cd16581b578 115 case Status::TestNotPerformed:
IanBenzMaxim 7:9cd16581b578 116 case Status::EntropyHealthy:
IanBenzMaxim 7:9cd16581b578 117 case Status::EntropyNotHealthy:
IanBenzMaxim 7:9cd16581b578 118 status.entropyHealthTestStatus =
IanBenzMaxim 7:9cd16581b578 119 static_cast<Status::EntropyHealthTestStatus>(*responseIt);
IanBenzMaxim 7:9cd16581b578 120 break;
IanBenzMaxim 7:9cd16581b578 121
IanBenzMaxim 7:9cd16581b578 122 default:
IanBenzMaxim 7:9cd16581b578 123 result = make_error_code(InvalidResponseError);
IanBenzMaxim 7:9cd16581b578 124 break;
IanBenzMaxim 7:9cd16581b578 125 }
IanBenzMaxim 7:9cd16581b578 126 }
IanBenzMaxim 7:9cd16581b578 127 return result;
IanBenzMaxim 7:9cd16581b578 128 }
IanBenzMaxim 7:9cd16581b578 129
IanBenzMaxim 7:9cd16581b578 130 error_code DS28E39::setPageProtection(int pageNum,
IanBenzMaxim 7:9cd16581b578 131 const PageProtection & protection) {
IanBenzMaxim 7:9cd16581b578 132 if (pageNum < 0 || pageNum >= memoryPages) {
IanBenzMaxim 7:9cd16581b578 133 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 134 }
IanBenzMaxim 7:9cd16581b578 135
IanBenzMaxim 7:9cd16581b578 136 int delay = writeStateTimeMs;
IanBenzMaxim 7:9cd16581b578 137 if (pageNum == decrementCounterPage) {
IanBenzMaxim 7:9cd16581b578 138 delay += writeMemoryTimeMs;
IanBenzMaxim 7:9cd16581b578 139 }
IanBenzMaxim 7:9cd16581b578 140 const uint_least8_t request[] = {
IanBenzMaxim 7:9cd16581b578 141 0xC3, static_cast<uint_least8_t>(pageNum),
IanBenzMaxim 7:9cd16581b578 142 static_cast<uint_least8_t>(protection.to_ulong())};
IanBenzMaxim 7:9cd16581b578 143 return runCommand(request, delay);
IanBenzMaxim 7:9cd16581b578 144 }
IanBenzMaxim 7:9cd16581b578 145
IanBenzMaxim 7:9cd16581b578 146 error_code
IanBenzMaxim 7:9cd16581b578 147 DS28E39::computeAndReadPageAuthentication(int pageNum, bool anonymous,
IanBenzMaxim 7:9cd16581b578 148 Page::const_span challenge,
IanBenzMaxim 7:9cd16581b578 149 Ecc256::Signature::span signature) {
IanBenzMaxim 7:9cd16581b578 150 if (pageNum < 0 || pageNum >= memoryPages) {
IanBenzMaxim 7:9cd16581b578 151 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 152 }
IanBenzMaxim 7:9cd16581b578 153
IanBenzMaxim 7:9cd16581b578 154 const size_t requestSize = 2 + Page::size;
IanBenzMaxim 7:9cd16581b578 155 const size_t responseSize = 1 + 2 * Ecc256::Scalar::size;
IanBenzMaxim 7:9cd16581b578 156 uint_least8_t buffer[MaximInterfaceCore_MAX(requestSize, responseSize)];
IanBenzMaxim 7:9cd16581b578 157 buffer[0] = 0xA5;
IanBenzMaxim 7:9cd16581b578 158 buffer[1] = pageNum | (anonymous ? 0xE0 : 0x00);
IanBenzMaxim 7:9cd16581b578 159 copy(challenge.begin(), challenge.end(), buffer + 2);
IanBenzMaxim 7:9cd16581b578 160 span<uint_least8_t> response(buffer, responseSize);
IanBenzMaxim 7:9cd16581b578 161 const error_code result = runCommand(make_span(buffer, requestSize),
IanBenzMaxim 7:9cd16581b578 162 generateEcdsaSignatureTimeMs, response);
IanBenzMaxim 7:9cd16581b578 163 if (!result) {
IanBenzMaxim 7:9cd16581b578 164 span<uint_least8_t>::const_iterator begin = response.begin();
IanBenzMaxim 7:9cd16581b578 165 span<uint_least8_t>::const_iterator end = begin + signature.s.size();
IanBenzMaxim 7:9cd16581b578 166 copy(begin, end, signature.s.begin());
IanBenzMaxim 7:9cd16581b578 167 begin = end;
IanBenzMaxim 7:9cd16581b578 168 end = begin + signature.r.size();
IanBenzMaxim 7:9cd16581b578 169 copy(begin, end, signature.r.begin());
IanBenzMaxim 7:9cd16581b578 170 }
IanBenzMaxim 7:9cd16581b578 171 return result;
IanBenzMaxim 7:9cd16581b578 172 }
IanBenzMaxim 7:9cd16581b578 173
IanBenzMaxim 7:9cd16581b578 174 error_code DS28E39::decrementCounter() {
IanBenzMaxim 7:9cd16581b578 175 const uint_least8_t request = 0xC9;
IanBenzMaxim 7:9cd16581b578 176 return runCommand(make_span(&request, 1), writeMemoryTimeMs);
IanBenzMaxim 7:9cd16581b578 177 }
IanBenzMaxim 7:9cd16581b578 178
IanBenzMaxim 7:9cd16581b578 179 error_code DS28E39::disableDevice() {
IanBenzMaxim 7:9cd16581b578 180 const uint_least8_t request[] = {0x33, 0x9E, 0xA7, 0x49, 0xFB,
IanBenzMaxim 7:9cd16581b578 181 0x10, 0x62, 0x0A, 0x26};
IanBenzMaxim 7:9cd16581b578 182 return runCommand(request, writeStateTimeMs);
IanBenzMaxim 7:9cd16581b578 183 }
IanBenzMaxim 7:9cd16581b578 184
IanBenzMaxim 7:9cd16581b578 185 error_code
IanBenzMaxim 7:9cd16581b578 186 DS28E39::readDevicePublicKey(Ecc256::PublicKey::span devicePublicKey) {
IanBenzMaxim 7:9cd16581b578 187 uint_least8_t buffer[1 + 2 * Ecc256::Scalar::size];
IanBenzMaxim 7:9cd16581b578 188 buffer[0] = 0xCB;
IanBenzMaxim 7:9cd16581b578 189 span<uint_least8_t> response(buffer);
IanBenzMaxim 7:9cd16581b578 190 const error_code result =
IanBenzMaxim 7:9cd16581b578 191 runCommand(make_span(buffer, 1), generateEccKeyPairTimeMs, response);
IanBenzMaxim 7:9cd16581b578 192 if (!result) {
IanBenzMaxim 7:9cd16581b578 193 span<uint_least8_t>::const_iterator begin = response.begin();
IanBenzMaxim 7:9cd16581b578 194 span<uint_least8_t>::const_iterator end = begin + devicePublicKey.x.size();
IanBenzMaxim 7:9cd16581b578 195 copy(begin, end, devicePublicKey.x.begin());
IanBenzMaxim 7:9cd16581b578 196 begin = end;
IanBenzMaxim 7:9cd16581b578 197 end = begin + devicePublicKey.y.size();
IanBenzMaxim 7:9cd16581b578 198 copy(begin, end, devicePublicKey.y.begin());
IanBenzMaxim 7:9cd16581b578 199 }
IanBenzMaxim 7:9cd16581b578 200 return result;
IanBenzMaxim 7:9cd16581b578 201 }
IanBenzMaxim 7:9cd16581b578 202
IanBenzMaxim 7:9cd16581b578 203 error_code DS28E39::readRng(span<uint_least8_t> data) {
IanBenzMaxim 7:9cd16581b578 204 const span<uint_least8_t>::index_type maxDataSize = 64;
IanBenzMaxim 7:9cd16581b578 205 if ((data.size() < 1) || (data.size() > maxDataSize)) {
IanBenzMaxim 7:9cd16581b578 206 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 207 }
IanBenzMaxim 7:9cd16581b578 208
IanBenzMaxim 7:9cd16581b578 209 uint_least8_t buffer[1 + maxDataSize];
IanBenzMaxim 7:9cd16581b578 210 buffer[0] = 0xD2;
IanBenzMaxim 7:9cd16581b578 211 buffer[1] = static_cast<uint_least8_t>(data.size() - 1);
IanBenzMaxim 7:9cd16581b578 212 span<uint_least8_t> response(buffer, 1 + data.size());
IanBenzMaxim 7:9cd16581b578 213 const error_code result =
IanBenzMaxim 7:9cd16581b578 214 runCommand(make_span(buffer, 2), trngGenerationTimeMs, response);
IanBenzMaxim 7:9cd16581b578 215 if (!result) {
IanBenzMaxim 7:9cd16581b578 216 copy(response.begin(), response.end(), data.begin());
IanBenzMaxim 7:9cd16581b578 217 }
IanBenzMaxim 7:9cd16581b578 218 return result;
IanBenzMaxim 7:9cd16581b578 219 }
IanBenzMaxim 7:9cd16581b578 220
IanBenzMaxim 7:9cd16581b578 221 error_code
IanBenzMaxim 7:9cd16581b578 222 DS28E39::authenticatePublicKey(Ecc256::Signature::const_span certificate,
IanBenzMaxim 7:9cd16581b578 223 span<const uint_least8_t> customization) {
IanBenzMaxim 7:9cd16581b578 224 static const span<const uint_least8_t>::index_type maxCustomizationSize = 32;
IanBenzMaxim 7:9cd16581b578 225 static const span<const uint_least8_t>::index_type signatureSize =
IanBenzMaxim 7:9cd16581b578 226 2 * Ecc256::Scalar::size;
IanBenzMaxim 7:9cd16581b578 227
IanBenzMaxim 7:9cd16581b578 228 if (customization.size() < 1 || customization.size() > maxCustomizationSize) {
IanBenzMaxim 7:9cd16581b578 229 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 230 }
IanBenzMaxim 7:9cd16581b578 231
IanBenzMaxim 7:9cd16581b578 232 uint_least8_t request[1 + signatureSize + maxCustomizationSize];
IanBenzMaxim 7:9cd16581b578 233 uint_least8_t * requestIt = request;
IanBenzMaxim 7:9cd16581b578 234 *requestIt++ = 0x59;
IanBenzMaxim 7:9cd16581b578 235 requestIt = copy(certificate.r.begin(), certificate.r.end(), requestIt);
IanBenzMaxim 7:9cd16581b578 236 requestIt = copy(certificate.s.begin(), certificate.s.end(), requestIt);
IanBenzMaxim 7:9cd16581b578 237 requestIt = copy(customization.begin(), customization.end(), requestIt);
IanBenzMaxim 7:9cd16581b578 238 return runCommand(make_span(request, requestIt), verifyEcdsaSignatureTimeMs);
IanBenzMaxim 7:9cd16581b578 239 }
IanBenzMaxim 7:9cd16581b578 240
IanBenzMaxim 7:9cd16581b578 241 error_code
IanBenzMaxim 7:9cd16581b578 242 DS28E39::authenticatedWriteMemory(int pageNum, Page::const_span page,
IanBenzMaxim 7:9cd16581b578 243 Ecc256::Signature::const_span signature) {
IanBenzMaxim 7:9cd16581b578 244 if (pageNum < 0 || pageNum >= memoryPages) {
IanBenzMaxim 7:9cd16581b578 245 return make_error_code(InvalidParameterError);
IanBenzMaxim 7:9cd16581b578 246 }
IanBenzMaxim 7:9cd16581b578 247
IanBenzMaxim 7:9cd16581b578 248 uint_least8_t request[2 + Page::size + 2 * Ecc256::Scalar::size];
IanBenzMaxim 7:9cd16581b578 249 uint_least8_t * requestIt = request;
IanBenzMaxim 7:9cd16581b578 250 *requestIt++ = 0x89;
IanBenzMaxim 7:9cd16581b578 251 *requestIt++ = pageNum;
IanBenzMaxim 7:9cd16581b578 252 requestIt = copy(page.begin(), page.end(), requestIt);
IanBenzMaxim 7:9cd16581b578 253 requestIt = copy(signature.r.begin(), signature.r.end(), requestIt);
IanBenzMaxim 7:9cd16581b578 254 copy(signature.s.begin(), signature.s.end(), requestIt);
IanBenzMaxim 7:9cd16581b578 255 return runCommand(request, verifyEcdsaSignatureTimeMs + writeMemoryTimeMs);
IanBenzMaxim 7:9cd16581b578 256 }
IanBenzMaxim 7:9cd16581b578 257
IanBenzMaxim 7:9cd16581b578 258 error_code DS28E39::runCommand(span<const uint_least8_t> request, int delayTime,
IanBenzMaxim 7:9cd16581b578 259 span<uint_least8_t> & response) {
IanBenzMaxim 7:9cd16581b578 260 const span<const uint_least8_t>::index_type responseInputSize =
IanBenzMaxim 7:9cd16581b578 261 response.size();
IanBenzMaxim 7:9cd16581b578 262 error_code result = doRunCommand(request, delayTime, response);
IanBenzMaxim 7:9cd16581b578 263 if (result) {
IanBenzMaxim 7:9cd16581b578 264 return result;
IanBenzMaxim 7:9cd16581b578 265 }
IanBenzMaxim 7:9cd16581b578 266 if (response.empty()) {
IanBenzMaxim 7:9cd16581b578 267 return make_error_code(InvalidResponseError);
IanBenzMaxim 7:9cd16581b578 268 }
IanBenzMaxim 7:9cd16581b578 269 // Parse command result byte.
IanBenzMaxim 7:9cd16581b578 270 switch (response[0]) {
IanBenzMaxim 7:9cd16581b578 271 case 0xAA:
IanBenzMaxim 7:9cd16581b578 272 // Success response.
IanBenzMaxim 7:9cd16581b578 273 if (response.size() != responseInputSize) {
IanBenzMaxim 7:9cd16581b578 274 result = make_error_code(InvalidResponseError);
IanBenzMaxim 7:9cd16581b578 275 }
IanBenzMaxim 7:9cd16581b578 276 break;
IanBenzMaxim 7:9cd16581b578 277
IanBenzMaxim 7:9cd16581b578 278 case 0x00:
IanBenzMaxim 7:9cd16581b578 279 result = make_error_code(AuthenticationError);
IanBenzMaxim 7:9cd16581b578 280 break;
IanBenzMaxim 7:9cd16581b578 281
IanBenzMaxim 7:9cd16581b578 282 default:
IanBenzMaxim 7:9cd16581b578 283 result.assign(response[0], errorCategory());
IanBenzMaxim 7:9cd16581b578 284 break;
IanBenzMaxim 7:9cd16581b578 285 }
IanBenzMaxim 7:9cd16581b578 286 response = response.subspan(1);
IanBenzMaxim 7:9cd16581b578 287 return result;
IanBenzMaxim 7:9cd16581b578 288 }
IanBenzMaxim 7:9cd16581b578 289
IanBenzMaxim 7:9cd16581b578 290 error_code DS28E39::runCommand(span<const uint_least8_t> request,
IanBenzMaxim 7:9cd16581b578 291 int delayTime) {
IanBenzMaxim 7:9cd16581b578 292 uint_least8_t buffer;
IanBenzMaxim 7:9cd16581b578 293 span<uint_least8_t> response(&buffer, 1);
IanBenzMaxim 7:9cd16581b578 294 return runCommand(request, delayTime, response);
IanBenzMaxim 7:9cd16581b578 295 }
IanBenzMaxim 7:9cd16581b578 296
IanBenzMaxim 7:9cd16581b578 297 const error_category & DS28E39::errorCategory() {
IanBenzMaxim 7:9cd16581b578 298 static class : public error_category {
IanBenzMaxim 7:9cd16581b578 299 public:
IanBenzMaxim 7:9cd16581b578 300 virtual const char * name() const { return "DS28E39"; }
IanBenzMaxim 7:9cd16581b578 301
IanBenzMaxim 7:9cd16581b578 302 virtual std::string message(int condition) const {
IanBenzMaxim 7:9cd16581b578 303 switch (condition) {
IanBenzMaxim 7:9cd16581b578 304 case InvalidOperationError:
IanBenzMaxim 7:9cd16581b578 305 return "Invalid Operation Error";
IanBenzMaxim 7:9cd16581b578 306
IanBenzMaxim 7:9cd16581b578 307 case InvalidParameterError:
IanBenzMaxim 7:9cd16581b578 308 return "Invalid Parameter Error";
IanBenzMaxim 7:9cd16581b578 309
IanBenzMaxim 7:9cd16581b578 310 case InvalidSequenceError:
IanBenzMaxim 7:9cd16581b578 311 return "Invalid Sequence Error";
IanBenzMaxim 7:9cd16581b578 312
IanBenzMaxim 7:9cd16581b578 313 case InternalError:
IanBenzMaxim 7:9cd16581b578 314 return "Internal Error";
IanBenzMaxim 7:9cd16581b578 315
IanBenzMaxim 7:9cd16581b578 316 case DeviceDisabledError:
IanBenzMaxim 7:9cd16581b578 317 return "Device Disabled Error";
IanBenzMaxim 7:9cd16581b578 318
IanBenzMaxim 7:9cd16581b578 319 case AuthenticationError:
IanBenzMaxim 7:9cd16581b578 320 return "Authentication Error";
IanBenzMaxim 7:9cd16581b578 321
IanBenzMaxim 7:9cd16581b578 322 case InvalidResponseError:
IanBenzMaxim 7:9cd16581b578 323 return "Invalid Response Error";
IanBenzMaxim 7:9cd16581b578 324 }
IanBenzMaxim 7:9cd16581b578 325 return defaultErrorMessage(condition);
IanBenzMaxim 7:9cd16581b578 326 }
IanBenzMaxim 7:9cd16581b578 327 } instance;
IanBenzMaxim 7:9cd16581b578 328 return instance;
IanBenzMaxim 7:9cd16581b578 329 }
IanBenzMaxim 7:9cd16581b578 330
IanBenzMaxim 7:9cd16581b578 331 error_code readManId(DS28E39 & ds28e39, ManId::span manId) {
IanBenzMaxim 7:9cd16581b578 332 DS28E39::Status status;
IanBenzMaxim 7:9cd16581b578 333 const error_code result = ds28e39.readStatus(false, status);
IanBenzMaxim 7:9cd16581b578 334 if (!result) {
IanBenzMaxim 7:9cd16581b578 335 copy(make_span(status.manId), manId);
IanBenzMaxim 7:9cd16581b578 336 }
IanBenzMaxim 7:9cd16581b578 337 return result;
IanBenzMaxim 7:9cd16581b578 338 }
IanBenzMaxim 7:9cd16581b578 339
IanBenzMaxim 7:9cd16581b578 340 DS28E39::PageAuthenticationData &
IanBenzMaxim 7:9cd16581b578 341 DS28E39::PageAuthenticationData::setAnonymousRomId() {
IanBenzMaxim 7:9cd16581b578 342 std::fill(romId().begin(), romId().end(), 0xFF);
IanBenzMaxim 7:9cd16581b578 343 return *this;
IanBenzMaxim 7:9cd16581b578 344 }
IanBenzMaxim 7:9cd16581b578 345
IanBenzMaxim 7:9cd16581b578 346 } // namespace MaximInterfaceDevices