aboutsummaryrefslogtreecommitdiff
path: root/mesonbuild/minit.py
diff options
context:
space:
mode:
authorMichael Hirsch, Ph.D <10931741+scivision@users.noreply.github.com>2019-11-02 17:34:58 -0400
committerJussi Pakkanen <jpakkane@gmail.com>2019-11-02 23:34:58 +0200
commitbbe6db08e39179191f60e397b8837b5a685760d1 (patch)
treec08d488c46f16bb026c6570a3b957bee14aa94a5 /mesonbuild/minit.py
parentb3fe4a0a18edbd85c3a9e119912c968960f99574 (diff)
downloadmeson-bbe6db08e39179191f60e397b8837b5a685760d1.zip
meson-bbe6db08e39179191f60e397b8837b5a685760d1.tar.gz
meson-bbe6db08e39179191f60e397b8837b5a685760d1.tar.bz2
Minit templates modularization
Diffstat (limited to 'mesonbuild/minit.py')
-rw-r--r--mesonbuild/minit.py406
1 files changed, 76 insertions, 330 deletions
diff --git a/mesonbuild/minit.py b/mesonbuild/minit.py
index 4ae0ae3..e89b4cf 100644
--- a/mesonbuild/minit.py
+++ b/mesonbuild/minit.py
@@ -14,241 +14,20 @@
"""Code that creates simple startup projects."""
-import os, sys, re, shutil, subprocess
+from pathlib import Path
+import re, shutil, subprocess
from glob import glob
from mesonbuild import mesonlib
from mesonbuild.environment import detect_ninja
-lib_h_template = '''#pragma once
-#if defined _WIN32 || defined __CYGWIN__
- #ifdef BUILDING_{utoken}
- #define {utoken}_PUBLIC __declspec(dllexport)
- #else
- #define {utoken}_PUBLIC __declspec(dllimport)
- #endif
-#else
- #ifdef BUILDING_{utoken}
- #define {utoken}_PUBLIC __attribute__ ((visibility ("default")))
- #else
- #define {utoken}_PUBLIC
- #endif
-#endif
+from mesonbuild.templates.ctemplates import (create_exe_c_sample, create_lib_c_sample)
+from mesonbuild.templates.cpptemplates import (create_exe_cpp_sample, create_lib_cpp_sample)
+from mesonbuild.templates.objctemplates import (create_exe_objc_sample, create_lib_objc_sample)
+from mesonbuild.templates.dlangtemplates import (create_exe_d_sample, create_lib_d_sample)
+from mesonbuild.templates.fortrantemplates import (create_exe_fortran_sample, create_lib_fortran_sample)
+from mesonbuild.templates.rusttemplates import (create_exe_rust_sample, create_lib_rust_sample)
-int {utoken}_PUBLIC {function_name}();
-
-'''
-
-lib_c_template = '''#include <{header_file}>
-
-/* This function will not be exported and is not
- * directly callable by users of this library.
- */
-int internal_function() {{
- return 0;
-}}
-
-int {function_name}() {{
- return internal_function();
-}}
-'''
-
-lib_c_test_template = '''#include <{header_file}>
-#include <stdio.h>
-
-int main(int argc, char **argv) {{
- if(argc != 1) {{
- printf("%s takes no arguments.\\n", argv[0]);
- return 1;
- }}
- return {function_name}();
-}}
-'''
-
-lib_c_meson_template = '''project('{project_name}', 'c',
- version : '{version}',
- default_options : ['warning_level=3'])
-
-# These arguments are only used to build the shared library
-# not the executables that use the library.
-lib_args = ['-DBUILDING_{utoken}']
-
-shlib = shared_library('{lib_name}', '{source_file}',
- install : true,
- c_args : lib_args,
- gnu_symbol_visibility : 'hidden',
-)
-
-test_exe = executable('{test_exe_name}', '{test_source_file}',
- link_with : shlib)
-test('{test_name}', test_exe)
-
-# Make this library usable as a Meson subproject.
-{ltoken}_dep = declare_dependency(
- include_directories: include_directories('.'),
- link_with : shlib)
-
-# Make this library usable from the system's
-# package manager.
-install_headers('{header_file}', subdir : '{header_dir}')
-
-pkg_mod = import('pkgconfig')
-pkg_mod.generate(
- name : '{project_name}',
- filebase : '{ltoken}',
- description : 'Meson sample project.',
- subdirs : '{header_dir}',
- libraries : shlib,
- version : '{version}',
-)
-'''
-
-hello_c_template = '''#include <stdio.h>
-
-#define PROJECT_NAME "{project_name}"
-
-int main(int argc, char **argv) {{
- if(argc != 1) {{
- printf("%s takes no arguments.\\n", argv[0]);
- return 1;
- }}
- printf("This is project %s.\\n", PROJECT_NAME);
- return 0;
-}}
-'''
-
-hello_c_meson_template = '''project('{project_name}', 'c',
- version : '{version}',
- default_options : ['warning_level=3'])
-
-exe = executable('{exe_name}', '{source_name}',
- install : true)
-
-test('basic', exe)
-'''
-
-hello_cpp_template = '''#include <iostream>
-
-#define PROJECT_NAME "{project_name}"
-
-int main(int argc, char **argv) {{
- if(argc != 1) {{
- std::cout << argv[0] << "takes no arguments.\\n";
- return 1;
- }}
- std::cout << "This is project " << PROJECT_NAME << ".\\n";
- return 0;
-}}
-'''
-
-hello_cpp_meson_template = '''project('{project_name}', 'cpp',
- version : '{version}',
- default_options : ['warning_level=3',
- 'cpp_std=c++14'])
-
-exe = executable('{exe_name}', '{source_name}',
- install : true)
-
-test('basic', exe)
-'''
-
-lib_hpp_template = '''#pragma once
-#if defined _WIN32 || defined __CYGWIN__
- #ifdef BUILDING_{utoken}
- #define {utoken}_PUBLIC __declspec(dllexport)
- #else
- #define {utoken}_PUBLIC __declspec(dllimport)
- #endif
-#else
- #ifdef BUILDING_{utoken}
- #define {utoken}_PUBLIC __attribute__ ((visibility ("default")))
- #else
- #define {utoken}_PUBLIC
- #endif
-#endif
-
-namespace {namespace} {{
-
-class {utoken}_PUBLIC {class_name} {{
-
-public:
- {class_name}();
- int get_number() const;
-
-private:
-
- int number;
-
-}};
-
-}}
-
-'''
-
-lib_cpp_template = '''#include <{header_file}>
-
-namespace {namespace} {{
-
-{class_name}::{class_name}() {{
- number = 6;
-}}
-
-int {class_name}::get_number() const {{
- return number;
-}}
-
-}}
-'''
-
-lib_cpp_test_template = '''#include <{header_file}>
-#include <iostream>
-
-int main(int argc, char **argv) {{
- if(argc != 1) {{
- std::cout << argv[0] << " takes no arguments.\\n";
- return 1;
- }}
- {namespace}::{class_name} c;
- return c.get_number() != 6;
-}}
-'''
-
-lib_cpp_meson_template = '''project('{project_name}', 'cpp',
- version : '{version}',
- default_options : ['warning_level=3', 'cpp_std=c++14'])
-
-# These arguments are only used to build the shared library
-# not the executables that use the library.
-lib_args = ['-DBUILDING_{utoken}']
-
-shlib = shared_library('{lib_name}', '{source_file}',
- install : true,
- cpp_args : lib_args,
- gnu_symbol_visibility : 'hidden',
-)
-
-test_exe = executable('{test_exe_name}', '{test_source_file}',
- link_with : shlib)
-test('{test_name}', test_exe)
-
-# Make this library usable as a Meson subproject.
-{ltoken}_dep = declare_dependency(
- include_directories: include_directories('.'),
- link_with : shlib)
-
-# Make this library usable from the system's
-# package manager.
-install_headers('{header_file}', subdir : '{header_dir}')
-
-pkg_mod = import('pkgconfig')
-pkg_mod.generate(
- name : '{project_name}',
- filebase : '{ltoken}',
- description : 'Meson sample project.',
- subdirs : '{header_dir}',
- libraries : shlib,
- version : '{version}',
-)
-'''
+FORTRAN_SUFFIXES = ['.f', '.for', '.F', '.f90', '.F90']
info_message = '''Sample project created. To build it run the
following commands:
@@ -257,76 +36,6 @@ meson builddir
ninja -C builddir
'''
-def create_exe_c_sample(project_name, project_version):
- lowercase_token = re.sub(r'[^a-z0-9]', '_', project_name.lower())
- source_name = lowercase_token + '.c'
- open(source_name, 'w').write(hello_c_template.format(project_name=project_name))
- open('meson.build', 'w').write(hello_c_meson_template.format(project_name=project_name,
- exe_name=lowercase_token,
- source_name=source_name,
- version=project_version))
-
-def create_lib_c_sample(project_name, version):
- lowercase_token = re.sub(r'[^a-z0-9]', '_', project_name.lower())
- uppercase_token = lowercase_token.upper()
- function_name = lowercase_token[0:3] + '_func'
- lib_h_name = lowercase_token + '.h'
- lib_c_name = lowercase_token + '.c'
- test_c_name = lowercase_token + '_test.c'
- kwargs = {'utoken': uppercase_token,
- 'ltoken': lowercase_token,
- 'header_dir': lowercase_token,
- 'function_name': function_name,
- 'header_file': lib_h_name,
- 'source_file': lib_c_name,
- 'test_source_file': test_c_name,
- 'test_exe_name': lowercase_token,
- 'project_name': project_name,
- 'lib_name': lowercase_token,
- 'test_name': lowercase_token,
- 'version': version,
- }
- open(lib_h_name, 'w').write(lib_h_template.format(**kwargs))
- open(lib_c_name, 'w').write(lib_c_template.format(**kwargs))
- open(test_c_name, 'w').write(lib_c_test_template.format(**kwargs))
- open('meson.build', 'w').write(lib_c_meson_template.format(**kwargs))
-
-def create_exe_cpp_sample(project_name, project_version):
- lowercase_token = re.sub(r'[^a-z0-9]', '_', project_name.lower())
- source_name = lowercase_token + '.cpp'
- open(source_name, 'w').write(hello_cpp_template.format(project_name=project_name))
- open('meson.build', 'w').write(hello_cpp_meson_template.format(project_name=project_name,
- exe_name=lowercase_token,
- source_name=source_name,
- version=project_version))
-
-def create_lib_cpp_sample(project_name, version):
- lowercase_token = re.sub(r'[^a-z0-9]', '_', project_name.lower())
- uppercase_token = lowercase_token.upper()
- class_name = uppercase_token[0] + lowercase_token[1:]
- namespace = lowercase_token
- lib_h_name = lowercase_token + '.hpp'
- lib_c_name = lowercase_token + '.cpp'
- test_c_name = lowercase_token + '_test.cpp'
- kwargs = {'utoken': uppercase_token,
- 'ltoken': lowercase_token,
- 'header_dir': lowercase_token,
- 'class_name': class_name,
- 'namespace': namespace,
- 'header_file': lib_h_name,
- 'source_file': lib_c_name,
- 'test_source_file': test_c_name,
- 'test_exe_name': lowercase_token,
- 'project_name': project_name,
- 'lib_name': lowercase_token,
- 'test_name': lowercase_token,
- 'version': version,
- }
- open(lib_h_name, 'w').write(lib_hpp_template.format(**kwargs))
- open(lib_c_name, 'w').write(lib_cpp_template.format(**kwargs))
- open(test_c_name, 'w').write(lib_cpp_test_template.format(**kwargs))
- open('meson.build', 'w').write(lib_cpp_meson_template.format(**kwargs))
-
def create_sample(options):
if options.language == 'c':
if options.type == 'executable':
@@ -342,17 +51,44 @@ def create_sample(options):
create_lib_cpp_sample(options.name, options.version)
else:
raise RuntimeError('Unreachable code')
+ elif options.language == 'd':
+ if options.type == 'executable':
+ create_exe_d_sample(options.name, options.version)
+ elif options.type == 'library':
+ create_lib_d_sample(options.name, options.version)
+ else:
+ raise RuntimeError('Unreachable code')
+ elif options.language == 'fortran':
+ if options.type == 'executable':
+ create_exe_fortran_sample(options.name, options.version)
+ elif options.type == 'library':
+ create_lib_fortran_sample(options.name, options.version)
+ else:
+ raise RuntimeError('Unreachable code')
+ elif options.language == 'rust':
+ if options.type == 'executable':
+ create_exe_rust_sample(options.name, options.version)
+ elif options.type == 'library':
+ create_lib_rust_sample(options.name, options.version)
+ else:
+ raise RuntimeError('Unreachable code')
+ elif options.language == 'objc':
+ if options.type == 'executable':
+ create_exe_objc_sample(options.name, options.version)
+ elif options.type == 'library':
+ create_lib_objc_sample(options.name, options.version)
+ else:
+ raise RuntimeError('Unreachable code')
else:
raise RuntimeError('Unreachable code')
print(info_message)
def autodetect_options(options, sample=False):
if not options.name:
- options.name = os.path.basename(os.getcwd())
+ options.name = Path().resolve().stem
if not re.match('[a-zA-Z_][a-zA-Z0-9]*', options.name) and sample:
- print('Name of current directory "{}" is not usable as a sample project name.\n'
- 'Specify a project name with --name.'.format(options.name))
- sys.exit(1)
+ raise SystemExit('Name of current directory "{}" is not usable as a sample project name.\n'
+ 'Specify a project name with --name.'.format(options.name))
print('Using "{}" (name of current directory) as project name.'
.format(options.name))
if not options.executable:
@@ -364,28 +100,40 @@ def autodetect_options(options, sample=False):
return
if not options.srcfiles:
srcfiles = []
- for f in os.listdir():
- if f.endswith('.cc') or f.endswith('.cpp') or f.endswith('.c'):
+ for f in (f for f in Path().iterdir() if f.is_file()):
+ if f.suffix in (['.cc', '.cpp', '.c', '.d', '.m', '.rs'] + FORTRAN_SUFFIXES):
srcfiles.append(f)
if not srcfiles:
- print("No recognizable source files found.\n"
- "Run me in an empty directory to create a sample project.")
- sys.exit(1)
+ raise SystemExit('No recognizable source files found.\n'
+ 'Run meson init in an empty directory to create a sample project.')
options.srcfiles = srcfiles
- print("Detected source files: " + ' '.join(srcfiles))
+ print("Detected source files: " + ' '.join(map(str, srcfiles)))
+ options.srcfiles = [Path(f) for f in options.srcfiles]
if not options.language:
for f in options.srcfiles:
- if f.endswith('.cc') or f.endswith('.cpp'):
+ if f.suffix in ('.cc', '.cpp'):
options.language = 'cpp'
break
- if f.endswith('.c'):
+ if f.suffix == '.c':
options.language = 'c'
break
+ if f.suffix == '.d':
+ options.language = 'd'
+ break
+ if f.suffix in FORTRAN_SUFFIXES:
+ options.language = 'fortran'
+ break
+ if f.suffix == '.rs':
+ options.language = 'rust'
+ break
+ if f.suffix == '.m':
+ options.language = 'objc'
+ break
if not options.language:
- print("Can't autodetect language, please specify it with -l.")
- sys.exit(1)
+ raise SystemExit("Can't autodetect language, please specify it with -l.")
print("Detected language: " + options.language)
+
meson_executable_template = '''project('{project_name}', '{language}',
version : '{version}',
default_options : [{default_options}])
@@ -397,10 +145,9 @@ executable('{executable}',
def create_meson_build(options):
if options.type != 'executable':
- print('\nGenerating a meson.build file from existing sources is\n'
- 'supported only for project type "executable".\n'
- 'Run me in an empty directory to create a sample project.')
- sys.exit(1)
+ raise SystemExit('\nGenerating a meson.build file from existing sources is\n'
+ 'supported only for project type "executable".\n'
+ 'Run meson init in an empty directory to create a sample project.')
default_options = ['warning_level=3']
if options.language == 'cpp':
# This shows how to set this very common option.
@@ -431,7 +178,7 @@ def add_arguments(parser):
parser.add_argument("-n", "--name", help="project name. default: name of current directory")
parser.add_argument("-e", "--executable", help="executable name. default: project name")
parser.add_argument("-d", "--deps", help="dependencies, comma-separated")
- parser.add_argument("-l", "--language", choices=['c', 'cpp'],
+ parser.add_argument("-l", "--language", choices=['c', 'cpp', 'd', 'fortran', 'rust', 'objc'],
help="project language. default: autodetected based on source files")
parser.add_argument("-b", "--build", help="build after generation", action='store_true')
parser.add_argument("--builddir", help="directory for build", default='build')
@@ -441,7 +188,7 @@ def add_arguments(parser):
choices=['executable', 'library'])
parser.add_argument('--version', default='0.1')
-def run(options):
+def run(options) -> int:
if not glob('*'):
autodetect_options(options, sample=True)
if not options.language:
@@ -450,21 +197,20 @@ def run(options):
create_sample(options)
else:
autodetect_options(options)
- if os.path.isfile('meson.build') and not options.force:
- print('meson.build already exists. Use --force to overwrite.')
- sys.exit(1)
+ if Path('meson.build').is_file() and not options.force:
+ raise SystemExit('meson.build already exists. Use --force to overwrite.')
create_meson_build(options)
if options.build:
- if os.path.isdir(options.builddir) and options.force:
+ if Path(options.builddir).is_dir() and options.force:
print('Build directory already exists, deleting it.')
shutil.rmtree(options.builddir)
print('Building...')
cmd = mesonlib.meson_command + [options.builddir]
- err = subprocess.call(cmd)
- if err:
- sys.exit(1)
+ ret = subprocess.run(cmd)
+ if ret.returncode:
+ raise SystemExit
cmd = [detect_ninja(), '-C', options.builddir]
- err = subprocess.call(cmd)
- if err:
- sys.exit(1)
+ ret = subprocess.run(cmd)
+ if ret.returncode:
+ raise SystemExit
return 0