Development through a C++ interpreter (Cling) as opposed to a standard compiler is an amazing leap in productivity and a window into the newest features of C++. This post tells you how to get your own bleeding-edge C++ interpreter built right on top of the development version of LLVM. We give you a repeatable procedure via Amazon EC2.
With our prescribed steps in place, you can always have an up-to-date development version of Cling. This allows quick testing and investigation of LLVM’s newest features. You will also possess a development environment as a path to contributing to Cling or learning how Cling’s innards push C++ code along.
Background
Astoundingly, there is such a thing as a C++ interpreter! Thus, if you’re interested in sticking to C++ for speed and efficiency, you’re not confined to the code-compile-run-debug workflow. Living outside such a workflow lends jumps in productivity and a unique angle through which to see C++.
Development of a serious C++ interpreter was undertaken by the wizards at CERN starting at around 1995. (You might recognize this group as searchers for the Higgs Boson). The C++ interpreter was dubbed CINT and was designed to support a major data analysis and visualization package known as ROOT (screenshot below). The system is certainly serious as it is giving researchers a way to investigate the very fabric of the universe!
Over the last 20 years, CINT fell behind the evolving C++ language. In response, individuals at CERN began work on a replacement for CINT dubbed Cling with the strategy of leveraging LLVM to stay up to date with the growing C++ specification. Up until recently, the project has been rather nascent, but now it is at the point where its functionality and build system make Cling worthwhile for use and contribution.
Build instructions for x86
We present a build procedure that leverages Amazon EC2. By using the cloud, you are guaranteed to use the exact Linux version and hardware capabilities that we did in testing. We assume that you have an Amazon EC2 account. The expected compute resource cost will be around two dollars.
-
Create an m3.xlarge instance with the Debian Jessie AMI (ami-aae18fc2). Be sure to grab at least 32Gb of storage.
-
Boot the instance and login as admin.
-
Install required build packages on the instance.
sudo apt-get update sudo apt-get install build-essential git
-
Use the distributed build script from the Cling project to clone required source code and build an x86 version of Cling with all available cores.
wget https://raw.githubusercontent.com/karies/cling-all-in-one/master/clone.sh chmod +x clone.sh ./clone.sh # will require a few minutes to finish
…this might take some time…
-
Take Cling for a spin!
Build instructions for ARM
Whereas an x86 build is readily supported by the Cling developers, building for ARM is not a standard build path. Thus, ARM build instructions are a bit more involved.
-
Log in as admin to the (same x86) instance described.
-
First, you’ll need LLVM and Clang as they provide utilities for compilation: llvm-tblgen and clang-tblgen.
sudo apt-get install llvm clang
-
Install ARM hard-float compilation tools.
sudo apt-get update # add Embedian as an installation source sudo bash -c 'echo deb http://emdebian.org/tools/debian/ jessie main > \ /etc/apt/sources.list.d/crosstools.list' sudo apt-get install -y curl curl http://emdebian.org/tools/debian/emdebian-toolchain-archive.key | sudo apt-key add - # add arm hard-float as a supported architecture sudo dpkg --add-architecture armhf sudo apt-get update # install an ARM targeted cross compiler sudo apt-get install crossbuild-essential-armhf
-
Install QEMU-based ARM emulation. This is a fantastic feature of newer linux kernels: ARM executables are registered with the system so that under a normal invocation, QEMU is invoked under the hood so that it seems as though ARM binaries are native.
# install QEMU emulation for ARM executables sudo apt-get install qemu binfmt-support qemu-user-static sudo update-binfmts --display sudo apt-get install libc6:armhf # compile an ARM executable cat > hello.c << EOF #include
int main(void) { return printf("Hello ARM!\n"); } EOF arm-linux-gnueabihf-gcc -static -ohello hello.c # the following should indicate an ARM exe file hello # running the executable with no fanfare should # transparently invoke QEMU to print "Hello ARM!" ./hello Having ARM emulation on our fast build machine makes debugging amazingly faster.
-
Install build tools: cmake and ninja.
# install cmake sudo apt-get install cmake # ninja requires a build git clone git://github.com/martine/ninja.git && cd ninja git checkout release ./configure.py --bootstrap chmod +x ninja build.ninja sudo cp ninja build.ninja /usr/bin
-
Clone LLVM, Clang, and Cling to the llvm directory. This directory differs from the instructions, above, so no worries if you’re following these steps in the order of the blog post. Note the specific SHA’s we jump to in the repositories. As of this writing, ARM support is in and out, but will hopefully materialize for good!
# the following steps assume this is done on a large volume # mounted at /mnt resulting in /mnt/llvm. git clone http://root.cern.ch/git/llvm.git cd llvm git checkout 88c56cd6d6aec0c44e9ca5aa96b354c1261a95aa cd tools git clone http://root.cern.ch/git/cling.git cd cling git checkout 899a5dde32cf670605a1478bea0b2d6fd3f95bef cd .. git clone http://root.cern.ch/git/clang.git cd clang git checkout d649bf1a04be728fc09222d111e696fce4b55324
-
Do a build with ninja and install to the install subdirectory. This will take a few minutes.
# from the /mnt/build directory mkdir -p install ninja install
-
Fire up Cling! You can do this directly on your Amazon EC2 instance through QEMU, but watch out – it is a tad unstable in the case of recovering from syntax errors, etc. However, you can copy what you built to an ARM machine such as a Raspberry Pi or Cubox to take it for a whirl there, too!
Configure a build in a new build directory using cmake. Note the intense and carefully crafted build flags required to do a successful cross compilation. This took quite a bit of debugging to lock down.
Note that the following assumes that the prior step was run in /mnt
resulting in an /mnt/llvm
directory. Indeed, the following block assumes that it is run in the /mnt
directory, as well.
mkdir -p build; cd build
cmake -G Ninja /mnt/llvm -DCMAKE_CROSSCOMPILING=True \
-DCMAKE_SYSTEM_NAME=Linux \
-DCMAKE_SYSTEM_VERSION=1 \
-DCMAKE_SYSTEM_PROCESSOR=armv7 \
-DCMAKE_C_COMPILER=arm-linux-gnueabihf-gcc \
-DCMAKE_CXX_COMPILER=arm-linux-gnueabihf-g++ \
-DCMAKE_INSTALL_PREFIX=/mnt/build/install \
-DLLVM_TABLEGEN=/usr/bin/llvm-tblgen-3.5 \
-DCLANG_TABLEGEN=/usr/bin/clang-tblgen-3.5 \
-DLLVM_DEFAULT_TARGET_TRIPLE=arm-linux-gnueabihf \
-DLLVM_HOST_TRIPLE=arm--linux-gnueabihf \
-DLLVM_TARGET_ARCH=ARM \
-DLLVM_TARGETS_TO_BUILD=ARM \
-DCMAKE_CXX_FLAGS="\
-mcpu=cortex-a9 \
-I/usr/include/arm-linux-gnueabihf/c++/4.9 \
-I/usr/lib/gcc/arm-linux-gnueabihf/4.9/include \
-mfloat-abi=hard"
cd /mnt/build/install/bin
./cling