How to use c++11 to program the Arduino?

2019-01-16 14:12发布

问题:

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 improvements, not things that require standard library changes.

回答1:

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),

  1. Locate the platform.txt file,
    • AVR architecture => {install path}\hardware\arduino\avr\platform.txt
    • SAM architecture => {install path}\hardware\arduino\sam\platform.txt
  2. Inside that file, you can change any flag, for instance,
    • compiler.c.flags for changing the default compilation flags for C++ files.
    • compiler.cpp.flags for changing the default compilation flags for C++ files.
  3. You can also change any of the "recipes" or compile patters, at the corresponding section of the configuration file, under the title "AVR/SAM compile patterns".
  4. After making the changes, you must restart the Arduino IDE, at least on version 1.5.7.

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.



回答2:

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).



回答3:

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


回答4:

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);
}
  1. Compile the program.
  2. Rename the original avr-g++.exe to avr-g++.orig.exe (or any other name).
  3. Create avr-g++.ini file where the first line is FULL path to the original program (e.g. D:\Arduino\hardware\tools\avr\bin\avr-g++.orig.exe) and add additional parameters, one per line, as desired.

You're done!

Example avr-g++.ini:

D:\Arduino\hardware\tools\avr\bin\avr-g++.orig.exe
-std=c++0x

Hope, that helps!



回答5:

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.