You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							131 lines
						
					
					
						
							5.8 KiB
						
					
					
				
			
		
		
	
	
							131 lines
						
					
					
						
							5.8 KiB
						
					
					
				| # -*- coding: future_fstrings -*-
 | |
| #
 | |
| # Copyright 2019 Gianluca Frison, Dimitris Kouzoupis, Robin Verschueren,
 | |
| # Andrea Zanelli, Niels van Duijkeren, Jonathan Frey, Tommaso Sartor,
 | |
| # Branimir Novoselnik, Rien Quirynen, Rezart Qelibari, Dang Doan,
 | |
| # Jonas Koenemann, Yutao Chen, Tobias Schöls, Jonas Schlagenhauf, Moritz Diehl
 | |
| #
 | |
| # This file is part of acados.
 | |
| #
 | |
| # The 2-Clause BSD License
 | |
| #
 | |
| # Redistribution and use in source and binary forms, with or without
 | |
| # modification, are permitted provided that the following conditions are met:
 | |
| #
 | |
| # 1. Redistributions of source code must retain the above copyright notice,
 | |
| # this list of conditions and the following disclaimer.
 | |
| #
 | |
| # 2. Redistributions in binary form must reproduce the above copyright notice,
 | |
| # this list of conditions and the following disclaimer in the documentation
 | |
| # and/or other materials provided with the distribution.
 | |
| #
 | |
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | |
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | |
| # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 | |
| # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 | |
| # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 | |
| # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 | |
| # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 | |
| # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 | |
| # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 | |
| # POSSIBILITY OF SUCH DAMAGE.;
 | |
| #
 | |
| 
 | |
| import os
 | |
| import sys
 | |
| from subprocess import DEVNULL, call, STDOUT
 | |
| 
 | |
| 
 | |
| class CMakeBuilder:
 | |
|     """
 | |
|     Class to work with the `CMake` build system.
 | |
|     """
 | |
|     def __init__(self):
 | |
|         self._source_dir = None  # private source directory, this is set to code_export_dir
 | |
|         self.build_dir = 'build'
 | |
|         self._build_dir = None  # private build directory, usually rendered to abspath(build_dir)
 | |
|         self.generator = None
 | |
|         """Defines the generator, options can be found via `cmake --help` under 'Generator'. Type: string. Linux default 'Unix Makefiles', Windows 'Visual Studio 15 2017 Win64'; default value: `None`."""
 | |
|         # set something for Windows
 | |
|         if os.name == 'nt':
 | |
|             self.generator = 'Visual Studio 15 2017 Win64'
 | |
|         self.build_targets = None
 | |
|         """A comma-separated list of the build targets, if `None` then all targets will be build; type: List of strings; default: `None`."""
 | |
|         self.options_on = None
 | |
|         """List of strings as CMake options which are translated to '-D Opt[0]=ON -D Opt[1]=ON ...'; default: `None`."""
 | |
| 
 | |
|     # Generate the command string for handling the cmake command.
 | |
|     def get_cmd1_cmake(self):
 | |
|         defines_str = ''
 | |
|         if self.options_on is not None:
 | |
|             defines_arr = [f' -D{opt}=ON' for opt in self.options_on]
 | |
|             defines_str = ' '.join(defines_arr)
 | |
|         generator_str = ''
 | |
|         if self.generator is not None:
 | |
|             generator_str = f' -G"{self.generator}"'
 | |
|         return f'cmake -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX="{self._source_dir}"{defines_str}{generator_str} -Wdev -S"{self._source_dir}" -B"{self._build_dir}"'
 | |
| 
 | |
|     # Generate the command string for handling the build.
 | |
|     def get_cmd2_build(self):
 | |
|         import multiprocessing
 | |
|         cmd = f'cmake --build "{self._build_dir}" --config Release -j{multiprocessing.cpu_count()}'
 | |
|         if self.build_targets is not None:
 | |
|             cmd += f' -t {self.build_targets}'
 | |
|         return cmd
 | |
| 
 | |
|     # Generate the command string for handling the install command.
 | |
|     def get_cmd3_install(self):
 | |
|         return f'cmake --install "{self._build_dir}"'
 | |
| 
 | |
|     def exec(self, code_export_directory, verbose=True):
 | |
|         """
 | |
|         Execute the compilation using `CMake` with the given settings.
 | |
|         :param code_export_directory: must be the absolute path to the directory where the code was exported to
 | |
|         """
 | |
|         if(os.path.isabs(code_export_directory) is False):
 | |
|             print(f'(W) the code export directory "{code_export_directory}" is not an absolute path!')
 | |
|         self._source_dir = code_export_directory
 | |
|         self._build_dir = os.path.abspath(self.build_dir)
 | |
|         try:
 | |
|             os.mkdir(self._build_dir)
 | |
|         except FileExistsError as e:
 | |
|             pass
 | |
| 
 | |
|         try:
 | |
|             os.chdir(self._build_dir)
 | |
|             cmd_str = self.get_cmd1_cmake()
 | |
|             print(f'call("{cmd_str})"')
 | |
|             retcode = call(
 | |
|                 cmd_str,
 | |
|                 shell=True,
 | |
|                 stdout=None if verbose else DEVNULL,
 | |
|                 stderr=None if verbose else STDOUT
 | |
|             )
 | |
|             if retcode != 0:
 | |
|                 raise RuntimeError(f'CMake command "{cmd_str}" was terminated by signal {retcode}')
 | |
|             cmd_str = self.get_cmd2_build()
 | |
|             print(f'call("{cmd_str}")')
 | |
|             retcode = call(
 | |
|                 cmd_str,
 | |
|                 shell=True,
 | |
|                 stdout=None if verbose else DEVNULL,
 | |
|                 stderr=None if verbose else STDOUT
 | |
|             )
 | |
|             if retcode != 0:
 | |
|                 raise RuntimeError(f'Build command "{cmd_str}" was terminated by signal {retcode}')
 | |
|             cmd_str = self.get_cmd3_install()
 | |
|             print(f'call("{cmd_str}")')
 | |
|             retcode = call(
 | |
|                 cmd_str,
 | |
|                 shell=True,
 | |
|                 stdout=None if verbose else DEVNULL,
 | |
|                 stderr=None if verbose else STDOUT
 | |
|             )
 | |
|             if retcode != 0:
 | |
|                 raise RuntimeError(f'Install command "{cmd_str}" was terminated by signal {retcode}')
 | |
|         except OSError as e:
 | |
|             print("Execution failed:", e, file=sys.stderr)
 | |
|         except Exception as e:
 | |
|             print("Execution failed:", e, file=sys.stderr)
 | |
|             exit(1)
 | |
| 
 |