support more boards

master
Hideaki Tai 5 years ago
parent b9415cc798
commit 1104708583

@ -7,22 +7,70 @@
// Packet Definition : https://art-net.org.uk/structure/streaming-packets/artdmx-packet-definition/
#include <Arduino.h>
#ifndef __AVR__
#include <array>
#include <vector>
#include <functional>
#else
#include "RingBuffer.h"
#if defined(ARDUINO_ARCH_AVR)\
|| defined(ARDUINO_ARCH_MEGAAVR)\
|| defined(ARDUINO_ARCH_SAM)\
|| defined(ARDUINO_ARCH_SAMD)\
|| defined(ARDUINO_spresense_ast)
#define ARDUINOOSC_DISABLE_STL
#endif
#if defined(ESP_PLATFORM)\
|| defined(ESP8266)\
|| defined(ARDUINO_AVR_UNO_WIFI_REV2)\
|| defined(ARDUINO_SAMD_MKRWIFI1010)\
|| defined(ARDUINO_SAMD_MKRVIDOR4000)\
|| defined(ARDUINO_SAMD_MKR1000)\
|| defined(ARDUINO_SAMD_NANO_33_IOT)
#define ARDUINOOSC_ENABLE_WIFI
#endif
#if defined(TEENSYDUINO)\
|| defined(ESP8266)\
|| defined(ARDUINOOSC_DISABLE_STL)
#define ARDUINOOSC_ENABLE_ETHER
#endif
#if !defined (ARDUINOOSC_ENABLE_WIFI)\
&& !defined (ARDUINOOSC_ENABLE_ETHER)\
&& !defined (ARDUINOOSC_DISABLE_STL)
#error THIS PLATFORM HAS NO WIFI OR ETHERNET OR NOT SUPPORTED ARCHITECTURE. PLEASE LET ME KNOW!
#endif
#ifdef ARDUINOOSC_ENABLE_WIFI
#ifdef ESP_PLATFORM
#include <WiFi.h>
#include <WiFiUdp.h>
#elif defined (ESP8266)
#include <ESP8266WiFi.h>
#include <WiFiUdp.h>
#elif defined (TEENSYDUINO) || defined (__AVR__)
#elif defined (ARDUINO_AVR_UNO_WIFI_REV2)\
|| defined(ARDUINO_SAMD_MKRWIFI1010)\
|| defined(ARDUINO_SAMD_MKRVIDOR4000)\
|| defined(ARDUINO_SAMD_NANO_33_IOT)
#include <SPI.h>
#include <WiFiNINA.h>
#include <WiFiUdp.h>
#elif defined (ARDUINO_SAMD_MKR1000)
#include <SPI.h>
#include <WiFi101.h>
#include <WiFiUdp.h>
#endif
#endif // ARDUINOOSC_ENABLE_WIFI
#ifdef ARDUINOOSC_ENABLE_ETHER
#include <Ethernet.h>
#include <EthernetUdp.h>
#include "util/TeensyDirtySTLErrorSolution/TeensyDirtySTLErrorSolution.h"
#endif // ARDUINOOSC_ENABLE_ETHER
#ifdef ARDUINOOSC_DISABLE_STL
#include "util/ArxContainer/ArxContainer.h"
#else
#include <array>
#include <vector>
#include <functional>
#endif
@ -102,7 +150,7 @@ namespace arduino
static constexpr uint8_t NUM_PIXELS_PER_UNIV { 170 };
#ifdef __AVR__
#ifdef ARDUINOOSC_DISABLE_STL
template <uint16_t SIZE>
class Array
{
@ -119,7 +167,7 @@ namespace arduino
template <typename S>
class Sender_
{
#ifdef __AVR__
#ifdef ARDUINOOSC_DISABLE_STL
Array<PACKET_SIZE> packet;
#else
std::array<uint8_t, PACKET_SIZE> packet;
@ -233,10 +281,11 @@ namespace arduino
template <typename S>
class Receiver_
{
#ifdef __AVR__
#ifdef ARDUINOOSC_DISABLE_STL
typedef void (*CallbackType)(uint8_t* data, uint16_t size);
struct Map { uint32_t universe; CallbackType func; };
using CallbackMap = RingBuffer<Map, 8>;
arx::vector<Map, 8> v;
using CallbackMap = arx::vector<Map, 8>;
Array<PACKET_SIZE> packet;
#else
using CallbackType = std::function<void(uint8_t* data, uint16_t size)>;
@ -412,11 +461,12 @@ namespace arduino
}
}
#if defined (ESP_PLATFORM) || defined (ESP8266)
using Artnet = arduino::artnet::Manager<WiFiUDP>;
using ArtnetSender = arduino::artnet::Sender<WiFiUDP>;
using ArtnetReceiver = arduino::artnet::Receiver<WiFiUDP>;
#elif defined (TEENSYDUINO) || defined (__AVR__)
#ifdef ARDUINOOSC_ENABLE_WIFI
using ArtnetWiFi = arduino::artnet::Manager<WiFiUDP>;
using ArtnetWiFiSender = arduino::artnet::Sender<WiFiUDP>;
using ArtnetWiFiReceiver = arduino::artnet::Receiver<WiFiUDP>;
#endif
#ifdef ARDUINOOSC_ENABLE_ETHER
using Artnet = arduino::artnet::Manager<EthernetUDP>;
using ArtnetSender = arduino::artnet::Sender<EthernetUDP>;
using ArtnetReceiver = arduino::artnet::Receiver<EthernetUDP>;

@ -7,7 +7,7 @@ const IPAddress ip(192, 168, 1, 201);
const IPAddress gateway(192, 168, 1, 1);
const IPAddress subnet(255, 255, 255, 0);
ArtnetReceiver artnet;
ArtnetWiFiReceiver artnet;
uint32_t universe1 = 1;
uint32_t universe2 = 2;

@ -7,7 +7,7 @@ const IPAddress ip(192, 168, 1, 201);
const IPAddress gateway(192, 168, 1, 1);
const IPAddress subnet(255, 255, 255, 0);
Artnet artnet;
ArtnetWiFi artnet;
const uint16_t size = 512;
uint8_t data[size];

@ -7,7 +7,7 @@ const IPAddress ip(192, 168, 1, 201);
const IPAddress gateway(192, 168, 1, 1);
const IPAddress subnet(255, 255, 255, 0);
ArtnetSender artnet;
ArtnetWiFiSender artnet;
uint32_t universe = 1;
const uint16_t size = 512;

@ -0,0 +1,553 @@
#pragma once
#ifndef ARX_RINGBUFFER_H
#define ARX_RINGBUFFER_H
#include <Arduino.h>
#if defined(ARDUINO_ARCH_AVR)\
|| defined(ARDUINO_ARCH_MEGAAVR)\
|| defined(ARDUINO_ARCH_SAM)\
|| defined(ARDUINO_ARCH_SAMD)\
|| defined(ARDUINO_spresense_ast)
#define ARX_CONTAINER_DISABLED
#endif
#ifdef ARX_CONTAINER_DISABLED
#ifndef ARX_VECTOR_DEFAULT_SIZE
#define ARX_VECTOR_DEFAULT_SIZE 16
#endif // ARX_VECTOR_DEFAULT_SIZE
#ifndef ARX_DEQUE_DEFAULT_SIZE
#define ARX_DEQUE_DEFAULT_SIZE 16
#endif // ARX_DEQUE_DEFAULT_SIZE
#ifndef ARX_MAP_DEFAULT_SIZE
#define ARX_MAP_DEFAULT_SIZE 16
#endif // ARX_MAP_DEFAULT_SIZE
#ifndef ARX_TYPE_TRAITS_INITIALIZER_LIST_DEFINED
#define ARX_TYPE_TRAITS_INITIALIZER_LIST_DEFINED
namespace std
{
template<class T>
class initializer_list
{
private:
const T* array;
size_t len;
initializer_list(const T* a, size_t l) : array(a), len(l) {}
public:
initializer_list() : array(nullptr), len(0) {}
size_t size() const { return len; }
const T *begin() const { return array; }
const T *end() const { return array + len; }
};
}
#endif // ARX_TYPE_TRAITS_INITIALIZER_LIST_DEFINED
namespace arx {
namespace container
{
namespace detail
{
template<class T>
T&& move(T& t){ return static_cast<T&&>(t); }
}
}
template<typename T, size_t N>
class RingBuffer
{
protected:
T queue_[N];
int head_ {0};
int tail_ {0};
public:
RingBuffer()
: queue_()
, head_ {0}
, tail_{0}
{
}
RingBuffer(std::initializer_list<T> lst)
: queue_()
, head_(0)
, tail_(0)
{
for (auto it = lst.begin() ; it != lst.end() ; ++it)
{
queue_[tail_] = *it;
++tail_;
}
}
// copy
explicit RingBuffer(const RingBuffer& r)
: queue_()
, head_(r.head_)
, tail_(r.tail_)
{
for (size_t i = 0; i < r.size(); ++i)
queue_[i] = r.queue_[i];
}
RingBuffer& operator= (const RingBuffer& r)
{
head_ = r.head_;
tail_ = r.tail_;
for (size_t i = 0; i < r.size(); ++i)
queue_[i] = r.queue_[i];
return *this;
}
// move
RingBuffer(RingBuffer&& r)
{
head_ = container::detail::move(r.head_);
tail_ = container::detail::move(r.tail_);
for (size_t i = 0; i < r.size(); ++i)
queue_[i] = container::detail::move(r.queue_[i]);
}
RingBuffer& operator= (RingBuffer&& r)
{
head_ = container::detail::move(r.head_);
tail_ = container::detail::move(r.tail_);
for (size_t i = 0; i < r.size(); ++i)
queue_[i] = container::detail::move(r.queue_[i]);
return *this;
}
virtual ~RingBuffer() {}
using iterator = T*;
using const_iterator = const T*;
size_t capacity() const { return N; };
size_t size() const { return tail_ - head_; }
inline const T* data() const { return &(get(head_)); }
T* data() { return &(get(head_)); }
bool empty() const { return tail_ == head_; }
void clear() { head_ = 0; tail_ = 0; }
void pop() { pop_front(); }
void pop_front()
{
if (size() == 0) return;
if (size() == 1) clear();
else head_++;
}
void pop_back()
{
if (size() == 0) return;
if (size() == 1) clear();
else tail_--;
}
void push(const T& data) { push_back(data); }
void push(T&& data) { push_back(data); }
void push_back(const T& data)
{
get(tail_++) = data;
if (size() > N) ++head_;
};
void push_back(T&& data)
{
get(tail_++) = data;
if (size() > N) ++head_;
};
void push_front(const T& data)
{
get(head_--) = data;
if (size() > N) --tail_;
};
void push_front(T&& data)
{
get(head_--) = data;
if (size() > N) --tail_;
};
void emplace(const T& data) { push(data); }
void emplace(T&& data) { push(data); }
void emplace_back(const T& data) { push_back(data); }
void emplace_back(T&& data) { push_back(data); }
const T& front() const { return get(head_); };
T& front() { return get(head_); };
const T& back() const { return get(tail_ - 1); }
T& back() { return get(tail_ - 1); }
const T& operator[] (size_t index) const { return get(head_ + (int)index); }
T& operator[] (size_t index) { return get(head_ + (int)index); }
iterator begin() { return ptr(head_); }
iterator end() { return (queue_ + tail_); }
const_iterator begin() const { return (const_iterator)ptr(head_); }
const_iterator end() const { return (const_iterator)(queue_ + tail_); }
iterator erase(iterator p)
{
if (p == end()) return p;
for (T* pos = p + 1; pos != end(); ++pos)
*(pos - 1) = *pos;
--tail_;
return p;
}
void resize(size_t sz)
{
size_t s = size();
if (sz > size())
{
for (size_t i = 0; i < sz - s; ++i) push(T());
}
else if (sz < size())
{
for (size_t i = 0; i < s - sz; ++i) pop();
}
}
void assign(const_iterator first, const_iterator end)
{
clear();
const char* p = first;
while (p != end) push(*p++);
}
void shrink_to_fit()
{
// dummy
}
void reserve(size_t n)
{
(void)n;
// dummy
}
void insert(const_iterator pos, const_iterator first, const_iterator last)
{
if (pos != end())
{
size_t sz = 0;
{
iterator it = (iterator)first;
for (; it != last; ++it) ++sz;
}
iterator it = end() + sz - 1;
for (int i = sz; i > 0; --i, --it)
{
*it = *(it - sz);
}
it = (iterator)pos;
for (size_t i = 0; i < sz; ++i)
{
*it = *(first + i);
}
}
else
{
iterator it = (iterator)first;
for (; it != last; ++it)
{
push_back(*it);
}
}
}
private:
T& get(const int i)
{
if (i >= 0) return queue_[i % N];
else return queue_[N - abs(i % N)];
}
const T& get(const int i) const
{
if (i >= 0) return queue_[i % N];
else return queue_[N - abs(i % N)];
}
T* ptr(const int i)
{
if (i >= 0) return (T*)(queue_ + i % N);
else return (T*)(queue_ + N - abs(i % N));
}
const T* ptr(const int i) const
{
if (i >= 0) return (T*)(queue_ + i % N);
else return (T*)(queue_ + N - abs(i % N));
}
};
template <typename T, size_t N>
bool operator== (const RingBuffer<T, N>& x, const RingBuffer<T, N>& y)
{
if (x.size() != y.size()) return false;
for (size_t i = 0; i < x.size(); ++i)
if (x[i] != y[i]) return false;
return true;
}
template <typename T, size_t N>
bool operator!= (const RingBuffer<T, N>& x, const RingBuffer<T, N>& y)
{
return !(x == y);
}
template <typename T, size_t N = ARX_VECTOR_DEFAULT_SIZE>
struct vector : public RingBuffer<T, N>
{
vector() : RingBuffer<T, N>() { }
vector(std::initializer_list<T> lst) : RingBuffer<T, N>(lst) { }
// copy
vector(const vector& r) : RingBuffer<T, N>(r) { }
vector& operator= (const vector& r)
{
RingBuffer<T, N>::operator=(r);
return *this;
}
// move
vector(vector&& r) : RingBuffer<T, N>(r) { }
vector& operator= (vector&& r)
{
RingBuffer<T, N>::operator=(r);
return *this;
}
virtual ~vector() {}
private:
using RingBuffer<T, N>::pop;
using RingBuffer<T, N>::pop_front;
using RingBuffer<T, N>::push;
using RingBuffer<T, N>::push_front;
using RingBuffer<T, N>::emplace;
};
template <typename T, size_t N = ARX_DEQUE_DEFAULT_SIZE>
struct deque : public RingBuffer<T, N>
{
deque() : RingBuffer<T, N>() { }
deque(std::initializer_list<T> lst) : RingBuffer<T, N>(lst) { }
// copy
deque(const deque& r) : RingBuffer<T, N>(r) { }
deque& operator= (const deque& r)
{
RingBuffer<T, N>::operator=(r);
return *this;
}
// move
deque(deque&& r) : RingBuffer<T, N>(r) { }
deque& operator= (deque&& r)
{
RingBuffer<T, N>::operator=(r);
return *this;
}
virtual ~deque() {}
private:
using RingBuffer<T, N>::capacity;
using RingBuffer<T, N>::pop;
using RingBuffer<T, N>::push;
using RingBuffer<T, N>::emplace;
using RingBuffer<T, N>::assign;
using RingBuffer<T, N>::begin;
using RingBuffer<T, N>::end;
};
template <class T1, class T2>
struct pair
{
T1 first;
T2 second;
};
template <class T1, class T2>
pair<T1, T2> make_pair(const T1& t1, const T2& t2)
{
return {t1, t2};
};
template <typename T1, typename T2>
bool operator== (const pair<T1, T2>& x, const pair<T1, T2>& y)
{
return (x.first == y.first) && (x.second == y.second);
}
template <typename T1, typename T2>
bool operator!= (const pair<T1, T2>& x, const pair<T1, T2>& y)
{
return !(x == y);
}
template <class Key, class T, size_t N = ARX_MAP_DEFAULT_SIZE>
struct map : public RingBuffer<pair<Key, T>, N>
{
using iterator = typename RingBuffer<pair<Key, T>, N>::iterator;
using const_iterator = typename RingBuffer<pair<Key, T>, N>::const_iterator;
map() : RingBuffer<pair<Key, T>, N>() { }
map(std::initializer_list<pair<Key, T>> lst) : RingBuffer<pair<Key, T>, N>(lst) { }
// copy
map(const map& r) : RingBuffer<pair<Key, T>, N>(r) { }
map& operator= (const map& r)
{
RingBuffer<pair<Key, T>, N>::operator=(r);
return *this;
}
// move
map(map&& r) : RingBuffer<T, N>(r) { }
map& operator= (map&& r)
{
RingBuffer<pair<Key, T>, N>::operator=(r);
return *this;
}
virtual ~map() {}
const_iterator find(const Key& key) const
{
for (size_t i = 0; i < this->size(); ++i)
{
const_iterator it = this->begin() + i;
if (key == it->first)
return it;
}
return this->end();
}
pair<iterator, bool> insert(const Key& key, const T& t)
{
bool b {false};
iterator it = find(key);
if (it == this->end())
{
this->push(make_pair(key, t));
b = true;
it = this->end() - 1;
}
return {it, b};
}
pair<iterator, bool> insert(const pair<Key, T>& p)
{
bool b {false};
const_iterator it = find(p.first);
if (it == this->end())
{
this->push(p);
b = true;
it = this->end() - 1;
}
return {(iterator)it, b};
}
pair<iterator, bool> emplace(const Key& key, const T& t)
{
return insert(key, t);
}
pair<iterator, bool> emplace(const pair<Key, T>& p)
{
return insert(p);
}
const T& at(const Key& key) const
{
// iterator it = find(key);
// if (it != this->end()) return it->second;
// return T();
return find(key)->second;
}
T& at(const Key& key)
{
// iterator it = find(key);
// if (it != this->end()) return it->second;
// return T();
return find(key)->second;
}
iterator erase(const iterator it)
{
iterator i = (iterator)find(it->first);
if (i != this->end())
{
return this->erase(i);
}
return this->end();
}
iterator erase(const size_t index)
{
if (index < this->size())
{
iterator it = this->begin() + index;
return this->erase(it);
}
return this->end();
}
const T& operator[] (const Key& key)
{
const_iterator it = find(key);
if (it != this->end()) return it->second;
insert(::arx::make_pair(key, T()));
return this->back().second;
}
private:
using RingBuffer<pair<Key, T>, N>::capacity;
using RingBuffer<pair<Key, T>, N>::data;
using RingBuffer<pair<Key, T>, N>::pop;
using RingBuffer<pair<Key, T>, N>::pop_front;
using RingBuffer<pair<Key, T>, N>::pop_back;
using RingBuffer<pair<Key, T>, N>::push;
using RingBuffer<pair<Key, T>, N>::push_back;
using RingBuffer<pair<Key, T>, N>::push_front;
using RingBuffer<pair<Key, T>, N>::emplace_back;
using RingBuffer<pair<Key, T>, N>::front;
using RingBuffer<pair<Key, T>, N>::back;
using RingBuffer<pair<Key, T>, N>::resize;
using RingBuffer<pair<Key, T>, N>::assign;
using RingBuffer<pair<Key, T>, N>::shrink_to_fit;
};
} // namespace arx
template<typename T, size_t N>
using ArxRingBuffer = arx::RingBuffer<T, N>;
#endif // ARX_CONTAINER_DISABLED
#endif // ARX_RINGBUFFER_H

@ -0,0 +1,21 @@
MIT License
Copyright (c) 2019 Hideaki Tai
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 THE
AUTHORS OR COPYRIGHT HOLDERS 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.

@ -0,0 +1,166 @@
# ArxContainer
C++ container-like classes (vector, map, etc.) for Arduino which cannot use STL
## Note
`ArxContainer` is C++ container-__like__ classes for Arduino.
All of the functions is not supported currently.
Detail of these containers are described in Detail section.
## Supported Container Types
- `vector`
- `map` (`pair`)
- `deque`
## Supported Boards
This library is currently enabled only if you use following architecture.
Please use C++ Standard Template Library for other boards.
- AVR (Uno, Nano, Mega, etc.)
- MEGAAVR (Uno WiFi, Nano Ecery, etc.)
- SAMD (Zero, MKR, M0, etc.)
- SPRESENSE
## Usage
### vector
```C++
// initialize with initializer_list
arx::vector<int> vs {1, 2, 3};
// add contents
for (size_t i = 4; i <= 5; ++i)
vs.push_back(i);
// index access
for (size_t i = 0; i < vs.size(); ++i)
Serial.println(vs[i]);
// range-based access
for (const auto& v : vs)
Serial.println(v);
```
### map
``` C++
// initialize with initializer_list
arx::map<String, int> mp {{"one", 1}, {"two", 2}};
// add contents
mp.insert("three", 3);
mp["four"] = 4;
// range based access
for (const auto& m : mp)
{
Serial.print("{");
Serial.print(m.first); Serial.print(",");
Serial.print(m.second);
Serial.println("}");
}
// key access
Serial.print("one = "); Serial.println(mp["one"]);
Serial.print("two = "); Serial.println(mp["two"]);
Serial.print("three = "); Serial.println(mp["three"]);
Serial.print("four = "); Serial.println(mp["four"]);
```
### deque
```C++
// initialize with initializer_list
arx::deque<int> dq {1, 2, 3};
// add contents
for (int i = 4; i <= 5; ++i)
dq.push_back(i);
// index access
for (int i = 0; i < dq.size(); ++i)
Serial.print(dq[i]);
```
## Detail
`ArxContainer` is C++ container-__like__ classes for Arduino.
This library is based on `arx::RingBuffer` and `arx::xxxx` is limited-size container.
`arx::RingBuffer` can be used as:
```C++
ArxRingBuffer<uint8_t, 4> buffer;
buffer.push(1);
buffer.push(2);
buffer.push(3);
for(size_t i = 0; i < buffer.size(); ++i)
Serial.println(buffer[i]);
buffer.pop();
for(auto& b : buffer)
Serial.println(b);
```
`arx::xxxx` are derived from `RingBuffer` and defined as:
``` C++
namespace arx {
template <typename T, size_t N = ARX_VECTOR_DEFAULT_SIZE>
struct vector : public RingBuffer<T, N>
template <class Key, class T, size_t N = ARX_MAP_DEFAULT_SIZE>
struct map : public RingBuffer<pair<Key, T>, N>
template <typename T, size_t N = ARX_DEQUE_DEFAULT_SIZE>
struct deque : public RingBuffer<T, N>
}
```
So range-based loop cannot be applyed to `arx::deque` (iterator is not continuous because it is based on `RingBuffer`).
### Manage Size Limit of Container
Global default size of container can be changed by defining these macros before `#include <ArxContainer.h>`.
``` C++
#define ARX_VECTOR_DEFAULT_SIZE XX // default: 16
#define ARX_MAP_DEFAULT_SIZE XX // default: 16
#define ARX_DEQUE_DEFAULT_SIZE XX // default: 16
```
Or you can change each container size by template argument.
``` C++
arx::vector<int, 3> vs;
arx::map<String, int, 4> ms;
arx::deque<int, 5> ds;
```
## Roadmap
This library will be updated if I want to use more container interfaces on supported boards shown above.
PRs are welcome!
## Used Inside of
- [Packetizer](https://github.com/hideakitai/Packetizer)
- [MsgPack](https://github.com/hideakitai/MsgPack)
- [MsgPacketizer](https://github.com/hideakitai/MsgPacketizer)
- [ArduinoOSC](https://github.com/hideakitai/ArduinoOSC)
## License
MIT

@ -0,0 +1,20 @@
{
"name": "ArxContainer",
"keywords": "ringbuffer, vector, deque, map",
"description": "C++ container-like classes (vector, map, etc.) for Arduino which cannot use STL",
"repository":
{
"type": "git",
"url": "https://github.com/hideakitai/ArxContainer.git"
},
"authors":
{
"name": "Hideaki Tai",
"url": "https://github.com/hideakitai",
"maintainer": true
},
"version": "0.3.6",
"license": "MIT",
"frameworks": "arduino",
"platforms": "*"
}

@ -0,0 +1,9 @@
name=ArxContainer
version=0.3.6
author=hideakitai
maintainer=hideakitai
sentence=C++ container-like classes (vector, map, etc.) for Arduino which cannot use STL
paragraph=C++ container-like classes (vector, map, etc.) for Arduino which cannot use STL
category=Data Storage
url=https://github.com/hideakitai
architectures=*

@ -0,0 +1,21 @@
MIT License
Copyright (c) 2020 Hideaki Tai
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 THE
AUTHORS OR COPYRIGHT HOLDERS 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.

@ -0,0 +1,57 @@
# TeensyDirtySTLErrorSolution
dirty solution to use STL in teensy
## Introduction
Sometimes we want to use STL on Teensy, but Teensy 3.x in Arduino IDE does not support STL by default. Though STL is enabled on Teensy 4.x, but you have to enable it manually on Teensy 3.x. If you use PlatformIO, you can also use STL on Teensy 3.x.
| | Teensy 4.x | Teensy 3.x |
| ----------- | ----------------- | ----------------- |
| PlatformIO | Almost Compilable | Almost Compilable |
| Arduino IDE | Almost Compilable | NOT Compilable |
## Enable STL in Arduino IDE
To enable STL on Teensy 3.x using Arduino IDE, add `-lstdc++` to `build flags`, which is defined in `boards.txt`. Many STL can be used by adding this flag. You have to do this procedure every time you upgrade the version of Teensyduino.
| OS | default location |
| ------- | --------------------------------------------------------------- |
| macOS | `Arduino.app/Contents/Java/hardware/teensy/avr/boards.txt` |
| Windows | `C:\Program Files (x86)\Arduino\hardware\teensy\avr\boards.txt` |
#### Before
``` boards.txt
teensy36.build.flags.libs=-larm_cortexM4lf_math -lm
teensy35.build.flags.libs=-larm_cortexM4lf_math -lm
teensy31.build.flags.libs=-larm_cortexM4l_math -lm
```
#### After
``` boards.txt
teensy36.build.flags.libs=-larm_cortexM4lf_math -lm -lstdc++
teensy35.build.flags.libs=-larm_cortexM4lf_math -lm -lstdc++
teensy31.build.flags.libs=-larm_cortexM4l_math -lm -lstdc++
```
## Do you still have compile error?
Yes, maybe you still have compilation error in both 4.x and 3.x for some STLs... Then you should add this library (header file) to your project. Some dirty solution I found is included in this header file.
``` C++
#include <TeensyDirtySTLErrorSolution.h>
```
## Warning
This method is not recommended because STL is not enabled by default... (if you know the reason, please let me know). I am not responsible for any trouble that may occur as a result of the use of this library. Please use at your own risk.
## License
MIT

@ -0,0 +1,65 @@
#pragma once
#include <Arduino.h>
#ifdef TEENSYDUINO
#ifndef TEENSYDUINO_DIRTY_STL_ERROR_SOLUTION
#define TEENSYDUINO_DIRTY_STL_ERROR_SOLUTION
#if defined(__MK20DX256__) || defined(__MK64FX512__) || defined(__MK66FX1M0__) // TEENSY 3.1/3.2, 3.5, 3.6
extern "C"
{
int _write() { return -1; }
}
#elif defined(__IMXRT1062__) // TEENSY4.0
extern "C"
{
void *__exidx_start __attribute__((__visibility__ ("hidden")));
void *__exidx_end __attribute__((__visibility__ ("hidden")));
}
#else
#error NOT SUPPORTED TEENSY VERSION
#endif
// ----- template -----
//
// #if defined(__MK20DX256__) // TEENSY3.1/3.2
// #elif defined(__MK64FX512__) // TEENSY3.5
// #elif defined(__MK66FX1M0__) // TEENSY3.6
// #elif defined(__IMXRT1062__) // TEENSY4.0
//
// extern "C"
// {
// int _getpid() { return -1; }
// int _kill(int pid, int sig) { return -1; }
// int _write() { return -1; }
// void *__exidx_start __attribute__((__visibility__ ("hidden")));
// void *__exidx_end __attribute__((__visibility__ ("hidden")));
// int _getpid();// { return -1; }
// int _kill(int pid, int sig);// { return -1; }
// int _write() { return -1; }
// void *__exidx_start __attribute__((__visibility__ ("hidden")));
// void *__exidx_end __attribute__((__visibility__ ("hidden")));
// }
//
// // copied from https://github.com/gcc-mirror/
// namespace std
// {
// void __throw_bad_alloc() { _GLIBCXX_THROW_OR_ABORT(bad_alloc()); }
// void __throw_length_error(const char* __s __attribute__((unused))) { _GLIBCXX_THROW_OR_ABORT(length_error(_(__s))); }
// void __throw_bad_function_call() { _GLIBCXX_THROW_OR_ABORT(bad_function_call()); }
// void _Rb_tree_decrement(std::_Rb_tree_node_base* a) {}
// void _Rb_tree_insert_and_rebalance(bool, std::_Rb_tree_node_base*, std::_Rb_tree_node_base*, std::_Rb_tree_node_base&) {}
// }
#endif // TEENSYDUINO_DIRTY_STL_ERROR_SOLUTION
#endif // TEENSYDUINO
Loading…
Cancel
Save