.. _getting-started-camerasdk-cpp-label: CameraSDK : C++ ~~~~~~~~~~~~~~~ The following section describe how to setup a CMake project, that works for both Windows and Linux based operating systems. Further, the CMake project automatically downloads the newest SDK binaries. Build & Runtime Prerequisites ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Depending on the target platform, our SDK's depends on different system libraries. Windows """"""" * Windows 10 (x64) * CMake 3.13 or later * MSVC 2019 or later On Windows CMake will output Visual Studio (``*.vcxproj``) project files. So, if you prefer to work in Visual Studio, you can do that. .. tip:: Visual Studio 2019 actually include (optional) built-in support for working directly with CMake files. This means there is no need to output ``vcxproj`` files at all. However, the way to configure projects and debug parameters are different from the approach used with normal Visual C++ projects. You can add the optional CMake component from the *Visual Studio Installer* application. Linux """"" * Ubuntu 20.04 (Focal) or later * CMake 3.13 or later * Either: *GCC 9.3 or later* or *Clang 10 or later* * ``libpthread`` * ``libatomic`` (*CameraSDK only*) * ``libusb-1.0-0`` (*CameraSDK only and optional*) You can of course use older Ubuntu's or other Linux distros. We use Ubuntu 20.04 here, because it has a later CMake than 3.13 built into its *apt-get* repositories. .. caution:: *Ubuntu 18.04 (Bionic Beaver)* has a too old CMake (3.10) as part of its default *apt-get* repositories. If you wish to use *Bionic*, you must install CMake directly from its website. You should use the same or a newer GCC or Clang version, to ensure the :term:`ABI`'s are compatible. .. note:: In contrast to our Windows releases, we provide only a *Release* build of our libraries on Linux. Since GCC / Clang allows linking between libraries built using *Debug* and *Release* configurations. Setup Your Project ^^^^^^^^^^^^^^^^^^ Create a project folder, and name it something like ``hello_world``. This folder will contain the entire sample project - including the downloaded SDK libraries. .. code:: sh $ mkdir hello_world $ cd hello_world Source Code ^^^^^^^^^^^ Now, let us add some example code that calls the CameraSDK. Create a new (empty) text file called ``main.cpp``. Open the file in your favorite text editor and insert this content: .. literalinclude:: ../../labels/cppCameraProjectTemplateSource :language: cpp This source code will query for any USB connected Phase One cameras. The connected cameras are listed in the console, before the program exits. .. caution:: **On Linux**: *CameraSDK* has a soft dependency on ``libusb``. This means it will be used, if it is found. If not found, there will be *no USB support* and no cameras will be found. This is a *runtime* dependency only! You do not have to link with ``libusb``, when your build your CameraSDK based applications. CameraSDK will automatically look for ``libusb`` upon first interaction with USB. CMake File ^^^^^^^^^^ Still inside the ``hello_world`` directory, create a new text file called ``CMakeLists.txt`` and insert this content: .. literalinclude:: ../../labels/cppCameraProjectTemplate :language: cmake :linenos: .. content-if:: :format: html Download the file :download:`here <../../labels/cppCameraProjectTemplate>` This is our CMake project file, that will automatically download the SDK libraries and handle compiler setup, like include directories, pre-defines and linker directives. In short, everything above line 50 is downloading *Camera- & ImageSDK* and making them available to CMake. The file will work for *Windows*, *Linux* and *Linux ARM64* systems. Below line 50, we define our executable (called ``hello_world``), and that it should link with static libraries for both Camera- & ImageSDK. .. note:: The source code in ``hello_world`` does not use any ImageSDK functionality. We link with it here, just for demonstration purposes. .. See :ref:`cmake-in-depth-label` where we examine this ``CMakeLists.txt`` in greater detail. Build & Run ^^^^^^^^^^^ By CMake convention, we should create a new directory called ``build``, inside our project folder. Therefore, ``cd`` into this new ``build`` directory, before running two ``cmake`` commands to configure CMake, and compile the project: .. code:: sh $ mkdir build $ cd build $ cmake .. $ cmake --build . The first ``cmake`` command tells CMake to generate a :term:`target build system` project, from our ``CMakeLists.txt`` file. The second ``cmake`` command triggers CMake to build the project using that target build system. .. tip:: **On Windows** the default target build system is *Visual Studio projects*. When you run ``cmake ..`` a ``.vcxproj`` is created for the executable in our project. Then the ``cmake --build`` command will trigger the *MSBuild* tool, to build the Visual Studio project. **On Linux** the default target build system is *Unix Makefiles*. When ``cmake ..`` is run, *Makefiles* are created from your CMake project file. The second command ``cmake --build`` simply triggers the ``make`` command. Lastly, let's run the ``hello_world`` application. If you have a camera connected via USB, you should see something similar to this: .. content-if:: :format: html .. code-tabs:: .. code-tab:: sh :title: Linux (sh) $ ./hello_world Probing for available cameras... Found 1 camera * IQ4 150MP (JD012345) $ .. code-tab:: powershell :title: Windows (PS) PS> .\\Debug\\hello_world.exe Probing for available cameras... Found 1 camera * IQ4 150MP (JD012345) PS> .. content-if:: :format: pdf **On Linux:** .. code:: sh $ ./hello_world Probing for available cameras... Found 1 camera * IQ4 150MP (JD012345) $ **On Windows:** .. code:: powershell PS> .\\Debug\\hello_world.exe Probing for available cameras... Found 1 camera * IQ4 150MP (JD012345) PS> Notice, that the location of the executable file differs between the *target build systems*. Visual Studio likes to position the binary inside a directory named after the compile configuration. .. hint:: **On Linux**: In case you have attached a camera to USB, and do not see any cameras listed: * Verify that you have setup *UDEV rules* for your user account, that allows access to USB. * Verify that you have installed ``libusb-1.0-0`` APT package. See more in the :ref:`linux-trouble-label` appendix. Using Dynamic Linking ^^^^^^^^^^^^^^^^^^^^^ Let's change our project to use the provided *dynamic library* files, instead of the static ones. This means we need to change our ``CMakeLists.txt`` file. We need to modify the lines 55 and 56 in our ``CMakeLists.txt`` file: .. code:: cmake target_link_libraries(hello_world PRIVATE CameraSDK::CameraSdkCpp ImageSDK::ImageSdkCpp ) We simply remove the ``Static`` postfix from the two lines. **On Linux**, we can now compile and run the executable with the dynamic linkage. Copy to build directory """"""""""""""""""""""" .. note:: **On Windows** we also need to make sure that the SDK's DLL files is available to the executable at runtime. This means we must add a post-build step to copy them to the build directory. Luckily, CMake can help with that too. **On Linux** CMake automatically builds our executable with an embedded :term:`RPATH`, that points to the SDK library files inside the build tree. This will ensure the executable works without any copying of files, as long as it is run inside the development setup. .. When you need to deploy your application, you need to use CMake's `install `_ feature, to remove the ``RPATH``. Or you can explicitly instruct CMake to not write ``RPATH``'s into the application binary, when building. *This step is needed only on Windows.* If we try to run our app, the :term:`dynamic linker` will fail at runtime, since it cannot locate the SDK ``.dll`` files. Windows' :term:`dynamic linker` will look for DLL's in the same directory as the running ``.exe`` file. This behavior is unique to Windows. We must first copy these files to the build destination directory. Add the following lines to ``CMakeLists.txt``, to the bottom of the file: .. literalinclude:: ../../labels/cppProjectTemplateDynamic :language: cmake This effectively adds a post-build step to the build process, copying the needed DLL files to the build directory - if needed. After setting up the post-build step, you can run the executable successfully again.