Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependents: mbed_DS28EC20_GPIO
Diff: Platforms/dotnet/Uart.cpp
- Revision:
- 0:f77ad7f72d04
diff -r 000000000000 -r f77ad7f72d04 Platforms/dotnet/Uart.cpp
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/Platforms/dotnet/Uart.cpp Mon Nov 06 14:39:18 2017 -0600
@@ -0,0 +1,218 @@
+/*******************************************************************************
+* Copyright (C) 2017 Maxim Integrated Products, Inc., All Rights Reserved.
+*
+* Permission is hereby granted, free of charge, to any person obtaining a
+* copy of this software and associated documentation files (the "Software"),
+* to deal in the Software without restriction, including without limitation
+* the rights to use, copy, modify, merge, publish, distribute, sublicense,
+* and/or sell copies of the Software, and to permit persons to whom the
+* Software is furnished to do so, subject to the following conditions:
+*
+* The above copyright notice and this permission notice shall be included
+* in all copies or substantial portions of the Software.
+*
+* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
+* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
+* IN NO EVENT SHALL MAXIM INTEGRATED BE LIABLE FOR ANY CLAIM, DAMAGES
+* OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
+* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
+* OTHER DEALINGS IN THE SOFTWARE.
+*
+* Except as contained in this notice, the name of Maxim Integrated
+* Products, Inc. shall not be used except as stated in the Maxim Integrated
+* Products, Inc. Branding Policy.
+*
+* The mere transfer of this software does not imply any licenses
+* of trade secrets, proprietary technology, copyrights, patents,
+* trademarks, maskwork rights, or any other form of intellectual
+* property whatsoever. Maxim Integrated Products, Inc. retains all
+* ownership rights.
+*******************************************************************************/
+
+#include <msclr/auto_gcroot.h>
+#include <msclr/marshal_cppstd.h>
+#include <MaximInterface/Utilities/Error.hpp>
+#include "ChangeSizeType.hpp"
+#include "Sleep.hpp"
+#include "Uart.hpp"
+
+using msclr::interop::marshal_as;
+using std::string;
+using namespace System;
+using System::IO::Ports::SerialPort;
+using System::Runtime::InteropServices::Marshal;
+
+namespace MaximInterface {
+namespace dotnet {
+
+template <typename Type, typename Input> static bool isType(Input in) {
+ return dynamic_cast<Type>(in) != nullptr;
+}
+
+template <typename Func>
+static error_code executeTryCatchOperation(Func tryOperation) {
+ return executeTryCatchOperation(tryOperation, [] {});
+}
+
+template <typename TryFunc, typename CatchFunc>
+static error_code executeTryCatchOperation(TryFunc tryOperation,
+ CatchFunc catchOperation) {
+ error_code result;
+ try {
+ tryOperation();
+ } catch (Exception ^ e) {
+ catchOperation();
+ if (isType<ArgumentException ^>(e)) {
+ result = make_error_code(Uart::ArgumentError);
+ } else if (isType<InvalidOperationException ^>(e)) {
+ result = make_error_code(Uart::InvalidOperationError);
+ } else if (isType<System::IO::IOException ^>(e)) {
+ result = make_error_code(Uart::IOError);
+ } else if (isType<UnauthorizedAccessException ^>(e)) {
+ result = make_error_code(Uart::UnauthorizedAccessError);
+ } else if (isType<TimeoutException ^>(e)) {
+ result = make_error_code(Uart::TimeoutError);
+ } else {
+ throw;
+ }
+ }
+ return result;
+}
+
+template <typename Func>
+static error_code executeIfConnected(const Uart & serial, Func operation) {
+ return serial.connected() ? operation()
+ : make_error_code(Uart::NotConnectedError);
+}
+
+struct Uart::Data {
+ msclr::auto_gcroot<SerialPort ^> port;
+};
+
+Uart::Uart() : data(std::make_unique<Data>()) {}
+
+Uart::~Uart() = default;
+
+Uart::Uart(Uart &&) noexcept = default;
+
+Uart & Uart::operator=(Uart &&) noexcept = default;
+
+error_code Uart::connect(const string & portName) {
+ data->port = gcnew SerialPort;
+ return executeTryCatchOperation(
+ [this, &portName] {
+ data->port->PortName = marshal_as<String ^>(portName);
+ data->port->DtrEnable = true;
+ data->port->Open();
+ },
+ [this] { data->port.reset(); });
+}
+
+void Uart::disconnect() { data->port.reset(); }
+
+bool Uart::connected() const { return data->port.get() != nullptr; }
+
+string Uart::portName() const {
+ return connected() ? marshal_as<string>(data->port->PortName) : "";
+}
+
+error_code Uart::setBaud(int_least32_t baud) {
+ return executeIfConnected(*this, [this, baud] {
+ return executeTryCatchOperation(
+ [this, baud] { data->port->BaudRate = baud; });
+ });
+}
+
+error_code Uart::sendBreak() {
+ return executeIfConnected(*this, [this] {
+ return executeTryCatchOperation([this] {
+ data->port->BreakState = true;
+ sleep(1);
+ data->port->BreakState = false;
+ });
+ });
+}
+
+error_code Uart::clearReadBuffer() {
+ return executeIfConnected(*this, [this] {
+ return executeTryCatchOperation([this] { data->port->ReadExisting(); });
+ });
+}
+
+error_code Uart::writeByte(uint_least8_t data) { return writeBlock(&data, 1); }
+
+error_code Uart::writeBlock(const uint_least8_t * data, size_t dataLen) {
+ return executeIfConnected(*this, [this, data, dataLen] {
+ return executeTryCatchOperation([this, data, dataLen] {
+ changeSizeType<int>(
+ [this](const uint_least8_t * dataChunk, int dataChunkSize) {
+ auto dataManaged = gcnew array<uint_least8_t>(dataChunkSize);
+ Marshal::Copy(
+ static_cast<IntPtr>(const_cast<uint_least8_t *>(dataChunk)),
+ dataManaged, 0, dataChunkSize);
+ this->data->port->Write(dataManaged, 0, dataChunkSize);
+ },
+ data, dataLen);
+ });
+ });
+}
+
+error_code Uart::readByte(uint_least8_t & data) {
+ return executeIfConnected(*this, [this, &data] {
+ return executeTryCatchOperation(
+ [this, &data] { data = this->data->port->ReadByte(); });
+ });
+}
+
+error_code Uart::readBlock(uint_least8_t * data, size_t dataLen) {
+ return executeIfConnected(*this, [this, data, dataLen] {
+ return executeTryCatchOperation([this, data, dataLen] {
+ changeSizeType<int>(
+ [this](uint_least8_t * dataChunk, int dataChunkSize) {
+ auto dataManaged = gcnew array<uint_least8_t>(dataChunkSize);
+ int bytesRead = 0;
+ do {
+ bytesRead += this->data->port->Read(dataManaged, bytesRead,
+ dataChunkSize - bytesRead);
+ } while (bytesRead < dataChunkSize);
+ Marshal::Copy(dataManaged, 0, static_cast<IntPtr>(dataChunk),
+ dataChunkSize);
+ },
+ data, dataLen);
+ });
+ });
+}
+
+const error_category & Uart::errorCategory() {
+ static class : public error_category {
+ public:
+ virtual const char * name() const { return "dotnet UART"; }
+
+ virtual string message(int condition) const {
+ switch (condition) {
+ case NotConnectedError:
+ return "Not Connected Error";
+
+ case ArgumentError:
+ return "Argument Error";
+
+ case InvalidOperationError:
+ return "Invalid Operation Error";
+
+ case IOError:
+ return "IO Error";
+
+ case UnauthorizedAccessError:
+ return "Unauthorized Access Error";
+
+ default:
+ return defaultErrorMessage(condition);
+ }
+ }
+ } instance;
+ return instance;
+}
+
+} // namespace dotnet
+} // namespace MaximInterface
\ No newline at end of file