How can I use c++11
when programming the Arduino? I would be fine using either the Arduino IDE or another environment. I am most interested in the core language
If you need more control and better IDE, I recommend using Sloeber Plugin for Eclipse or the Sloeber IDE itself. Creating more complicated code is much easier using this IDE. It also allows to add flags to compiler (c, c++ and linker). So to customize the compile, just right click on project and select Properties. In the Properties window, select Arduino->Compiler Option. This way you can add options to your build.
Firstly, only gcc
4.7 and above (and therefore avr-gcc
4.7 and above) support C++11
. So, check the versions installed with :
gcc --version
avr-gcc --version
If avr-gcc
is 4.7 or higher, then you may be able to use C++11
.
The Arduino IDE does not support custom compiler flags. This has been requested but has not yet been implemented.
So, you are left with having to use other environments or to compile your program directly from the command line.
In case, of compiling directly from the command line using avr-gcc
, you simply need to add an extra compiler flag for enabling C++11 support.
-std=c++11
For specific development environments, most would support editing of the compiler flags from the build options within the IDE. The above mentioned flag needs to be added to the list of flags for each environment.
C++0x
was the name of working draft of the C++11
standard. C++0x
support is available gcc
4.3 onwards. However, this is strictly experimental support so you cannot realiably expect C++11
features to be present. Here is the complete list of features available with the corresponding version of gcc
. The availability of features in avr-gcc
will be the same as whats available in the corresponding gcc
version.
The compiler flag for C++0x
is :
-std=c++0x
As of version 1.6.6, the Arduino IDE enables c++11 by default.
For older versions, read on:
It is very easy to change the flags for any element of the toolchain, including the assembler, compiler, linker or archiver.
Tested on the Arduino IDE version 1.5.7 (released on July 2014),
For instance,
To enable support for C++11 (C++0x), tested on Arduino IDE versions 1.5.7 and 1.5.8, you will simply add the flag "-std=gnu++11" at the end of the line starting with compiler.cpp.flags=".
It is expected that C++11 is enabled by default in the near future on the Arduino IDE. However, as of version 1.5.8 (Oct 2014) it is still not the case.
Arduino IDE 1.6.6 and newer have C++11 enabled by default (they have the compiler flag "-std=gnu++11" set in the platform.txt file).
I use Ino and this worked:
ino build -cppflags="-std=c++0x"
This generated a hex file at least 15k in size (that's with optimizations turned on), compared to about 5k for the standard build, which is a consideration for a poor little Atmega328. Might be okay for one of the microcontrollers with a lot more program space.
Please, note, that there is no easy way to specify additional flags from Arduino IDE or use other IDE (Eclipse, Code Blocks, etc) or command line.
As a hack, you can use a small proxy program (should be cross-platform):
//============================================================================
// Name : gcc-proxy.cpp
// Copyright : Use as you want
// Description : Based on http://stackoverflow.com/questions/5846934/how-to-pass-a-vector-to-execvp
//============================================================================
#include <unistd.h>
#include <iostream>
#include <vector>
#include <fstream>
using namespace std;
int main(int argc, char *argv[]) {
vector<string> arguments;
vector<const char*> aptrs;
// Additional options, one per line
ifstream cfg((string(argv[0]) + ".ini").c_str());
if (cfg.bad())
cerr << "Could not open ini file (you're using proxy for some reason, er?)" << endl;
string arg;
while (cfg) {
getline(cfg, arg);
if(arg == "\r" || arg == "\n")
continue;
arguments.push_back(arg);
}
for (const string& arg : arguments)
aptrs.push_back(arg.c_str());
for (int i = 1; i < argc; ++i)
aptrs.push_back(argv[i]);
// Add null pointer at the end, execvp expects NULL as last element
aptrs.push_back(nullptr);
// pass the vector's internal array to execvp
const char **command = &aptrs[0];
return execvp(command[0], command);
}
You're done!
Example avr-g++.ini:
D:\Arduino\hardware\tools\avr\bin\avr-g++.orig.exe
-std=c++0x
Hope, that helps!