cmake_minimum_required(VERSION 3.10) project(quickjs LANGUAGES C) include(CheckCCompilerFlag) include(GNUInstallDirs) set(CMAKE_C_VISIBILITY_PRESET hidden) set(CMAKE_C_STANDARD_REQUIRED ON) set(CMAKE_C_EXTENSIONS ON) set(CMAKE_C_STANDARD 11) if(NOT CMAKE_BUILD_TYPE) message(STATUS "No build type selected, default to Release") set(CMAKE_BUILD_TYPE "Release") endif() message(STATUS "Building in ${CMAKE_BUILD_TYPE} mode") message(STATUS "Building with ${CMAKE_C_COMPILER_ID} ${CMAKE_C_COMPILER_VERSION} on ${CMAKE_SYSTEM}") macro(xcheck_add_c_compiler_flag FLAG) string(REPLACE "-" "" FLAG_NO_HYPHEN ${FLAG}) check_c_compiler_flag(${FLAG} COMPILER_SUPPORTS_${FLAG_NO_HYPHEN}) if(COMPILER_SUPPORTS_${FLAG_NO_HYPHEN}) add_compile_options(${FLAG}) endif() endmacro() xcheck_add_c_compiler_flag(-Wall) if(NOT MSVC AND NOT IOS) xcheck_add_c_compiler_flag(-Werror) xcheck_add_c_compiler_flag(-Wextra) endif() xcheck_add_c_compiler_flag(-Wno-implicit-fallthrough) xcheck_add_c_compiler_flag(-Wno-sign-compare) xcheck_add_c_compiler_flag(-Wno-missing-field-initializers) xcheck_add_c_compiler_flag(-Wno-unused-parameter) xcheck_add_c_compiler_flag(-Wno-unused-but-set-variable) xcheck_add_c_compiler_flag(-Wno-array-bounds) xcheck_add_c_compiler_flag(-Wno-format-truncation) xcheck_add_c_compiler_flag(-funsigned-char) # ClangCL is command line compatible with MSVC, so 'MSVC' is set. if(MSVC) xcheck_add_c_compiler_flag(-Wno-unsafe-buffer-usage) xcheck_add_c_compiler_flag(-Wno-sign-conversion) xcheck_add_c_compiler_flag(-Wno-nonportable-system-include-path) xcheck_add_c_compiler_flag(-Wno-implicit-int-conversion) xcheck_add_c_compiler_flag(-Wno-shorten-64-to-32) xcheck_add_c_compiler_flag(-Wno-reserved-macro-identifier) xcheck_add_c_compiler_flag(-Wno-reserved-identifier) xcheck_add_c_compiler_flag(-Wdeprecated-declarations) xcheck_add_c_compiler_flag(/experimental:c11atomics) xcheck_add_c_compiler_flag(/wd4018) # -Wno-sign-conversion xcheck_add_c_compiler_flag(/wd4061) # -Wno-implicit-fallthrough xcheck_add_c_compiler_flag(/wd4100) # -Wno-unused-parameter xcheck_add_c_compiler_flag(/wd4200) # -Wno-zero-length-array xcheck_add_c_compiler_flag(/wd4242) # -Wno-shorten-64-to-32 xcheck_add_c_compiler_flag(/wd4244) # -Wno-shorten-64-to-32 xcheck_add_c_compiler_flag(/wd4245) # -Wno-sign-compare xcheck_add_c_compiler_flag(/wd4267) # -Wno-shorten-64-to-32 xcheck_add_c_compiler_flag(/wd4388) # -Wno-sign-compare xcheck_add_c_compiler_flag(/wd4389) # -Wno-sign-compare xcheck_add_c_compiler_flag(/wd4710) # Function not inlined xcheck_add_c_compiler_flag(/wd4711) # Function was inlined xcheck_add_c_compiler_flag(/wd4820) # Padding added after construct xcheck_add_c_compiler_flag(/wd4996) # -Wdeprecated-declarations xcheck_add_c_compiler_flag(/wd5045) # Compiler will insert Spectre mitigation for memory load if /Qspectre switch specified endif() # MacOS and GCC 11 or later need -Wno-maybe-uninitialized # https://github.com/quickjs-ng/quickjs/issues/453 if(APPLE AND CMAKE_C_COMPILER_ID STREQUAL "GNU" AND CMAKE_C_COMPILER_VERSION VERSION_GREATER_EQUAL 11) xcheck_add_c_compiler_flag(-Wno-maybe-uninitialized) endif() if(CMAKE_SYSTEM_NAME STREQUAL "WASI") add_compile_definitions( _WASI_EMULATED_PROCESS_CLOCKS _WASI_EMULATED_SIGNAL ) add_link_options( -lwasi-emulated-process-clocks -lwasi-emulated-signal ) endif() if(CMAKE_BUILD_TYPE MATCHES "Debug") add_compile_options(-O0) xcheck_add_c_compiler_flag(-ggdb) xcheck_add_c_compiler_flag(-fno-omit-frame-pointer) endif() macro(xoption OPTION_NAME OPTION_TEXT OPTION_DEFAULT) option(${OPTION_NAME} ${OPTION_TEXT} ${OPTION_DEFAULT}) if(DEFINED ENV{${OPTION_NAME}}) # Allow setting the option through an environment variable. set(${OPTION_NAME} $ENV{${OPTION_NAME}}) endif() if(${OPTION_NAME}) add_definitions(-D${OPTION_NAME}) endif() message(STATUS " ${OPTION_NAME}: ${${OPTION_NAME}}") endmacro() xoption(BUILD_SHARED_LIBS "Build a shared library" OFF) if(BUILD_SHARED_LIBS) message(STATUS "Building a shared library") endif() # note: CONFIG_TSAN is currently incompatible with the other sanitizers but we # don't explicitly check for that because who knows what the future will bring? # CONFIG_MSAN only works with clang at the time of writing; also not checked # for the same reason xoption(BUILD_EXAMPLES "Build examples" OFF) xoption(BUILD_STATIC_QJS_EXE "Build a static qjs executable" OFF) xoption(BUILD_CLI_WITH_MIMALLOC "Build the qjs executable with mimalloc" OFF) xoption(BUILD_CLI_WITH_STATIC_MIMALLOC "Build the qjs executable with mimalloc (statically linked)" OFF) xoption(CONFIG_ASAN "Enable AddressSanitizer (ASan)" OFF) xoption(CONFIG_MSAN "Enable MemorySanitizer (MSan)" OFF) xoption(CONFIG_TSAN "Enable ThreadSanitizer (TSan)" OFF) xoption(CONFIG_UBSAN "Enable UndefinedBehaviorSanitizer (UBSan)" OFF) if(CONFIG_ASAN) message(STATUS "Building with ASan") add_compile_options( -fsanitize=address -fno-sanitize-recover=all -fno-omit-frame-pointer ) add_link_options( -fsanitize=address -fno-sanitize-recover=all -fno-omit-frame-pointer ) endif() if(CONFIG_MSAN) message(STATUS "Building with MSan") add_compile_options( -fsanitize=memory -fno-sanitize-recover=all -fno-omit-frame-pointer ) add_link_options( -fsanitize=memory -fno-sanitize-recover=all -fno-omit-frame-pointer ) endif() if(CONFIG_TSAN) message(STATUS "Building with TSan") add_compile_options( -fsanitize=thread -fno-sanitize-recover=all -fno-omit-frame-pointer ) add_link_options( -fsanitize=thread -fno-sanitize-recover=all -fno-omit-frame-pointer ) endif() if(CONFIG_UBSAN) message(STATUS "Building with UBSan") # __has_feature(undefined_sanitizer) or __SANITIZE_UNDEFINED__ don't exist add_compile_definitions( __UBSAN__=1 ) add_compile_options( -fsanitize=undefined -fno-sanitize-recover=all -fno-omit-frame-pointer ) add_link_options( -fsanitize=undefined -fno-sanitize-recover=all -fno-omit-frame-pointer ) endif() # QuickJS library # xoption(BUILD_QJS_LIBC "Build standard library modules as part of the library" OFF) macro(add_qjs_libc_if_needed target) if(NOT BUILD_QJS_LIBC) target_sources(${target} PRIVATE quickjs-libc.c) endif() endmacro() set(qjs_sources cutils.c libbf.c libregexp.c libunicode.c quickjs.c ) if(BUILD_QJS_LIBC) list(APPEND qjs_sources quickjs-libc.c) endif() list(APPEND qjs_defines _GNU_SOURCE) if(WIN32) list(APPEND qjs_defines WIN32_LEAN_AND_MEAN _WIN32_WINNT=0x0602) endif() list(APPEND qjs_libs ${CMAKE_DL_LIBS}) find_package(Threads) if(NOT CMAKE_SYSTEM_NAME STREQUAL "WASI") list(APPEND qjs_libs ${CMAKE_THREAD_LIBS_INIT}) endif() if(NOT MSVC) list(APPEND qjs_libs m) endif() add_library(qjs ${qjs_sources}) target_compile_definitions(qjs PRIVATE ${qjs_defines}) target_include_directories(qjs PUBLIC $ $ ) target_link_libraries(qjs PUBLIC ${qjs_libs}) if(EMSCRIPTEN) add_executable(qjs_wasm ${qjs_sources}) target_link_options(qjs_wasm PRIVATE # in emscripten 3.x, this will be set to 16k which is too small for quickjs. #write sth. to force github rebuild -sSTACK_SIZE=2097152 # let it be 2m = 2 * 1024 * 1024 = 2097152, otherwise, stack overflow may be occured at bootstrap -sNO_INVOKE_RUN -sNO_EXIT_RUNTIME -sMODULARIZE # do not mess the global -sEXPORT_ES6 # export js file to morden es module -sEXPORT_NAME=getQuickJs # give a name -sTEXTDECODER=1 # it will be 2 if we use -Oz, and that will cause js -> c string convertion fail -sNO_DEFAULT_TO_CXX # this project is pure c project, no need for c plus plus handle -sEXPORTED_RUNTIME_METHODS=ccall,cwrap ) target_compile_definitions(qjs_wasm PRIVATE ${qjs_defines}) target_link_libraries(qjs_wasm m) endif() # QuickJS bytecode compiler # add_executable(qjsc qjsc.c ) add_qjs_libc_if_needed(qjsc) target_compile_definitions(qjsc PRIVATE ${qjs_defines}) target_link_libraries(qjsc qjs) # QuickJS CLI # add_executable(qjs_exe gen/repl.c qjs.c ) add_qjs_libc_if_needed(qjs_exe) set_target_properties(qjs_exe PROPERTIES OUTPUT_NAME "qjs" ) target_compile_definitions(qjs_exe PRIVATE ${qjs_defines}) target_link_libraries(qjs_exe qjs) if(BUILD_STATIC_QJS_EXE OR MINGW) target_link_options(qjs_exe PRIVATE -static) if(MINGW) target_link_options(qjs_exe PRIVATE -static-libgcc) endif() endif() if(NOT WIN32) set_target_properties(qjs_exe PROPERTIES ENABLE_EXPORTS TRUE) endif() if(BUILD_CLI_WITH_MIMALLOC OR BUILD_CLI_WITH_STATIC_MIMALLOC) find_package(mimalloc REQUIRED) # Upstream mimalloc doesn't provide a way to know if both libraries are supported. if(BUILD_CLI_WITH_STATIC_MIMALLOC) target_link_libraries(qjs_exe mimalloc-static) else() target_link_libraries(qjs_exe mimalloc) endif() endif() # Test262 runner # if(NOT EMSCRIPTEN) add_executable(run-test262 run-test262.c ) add_qjs_libc_if_needed(run-test262) target_compile_definitions(run-test262 PRIVATE ${qjs_defines}) target_link_libraries(run-test262 qjs) endif() # Unicode generator # add_executable(unicode_gen EXCLUDE_FROM_ALL cutils.c libunicode.c unicode_gen.c ) target_compile_definitions(unicode_gen PRIVATE ${qjs_defines}) add_executable(function_source gen/function_source.c ) add_qjs_libc_if_needed(function_source) target_include_directories(function_source PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) target_compile_definitions(function_source PRIVATE ${qjs_defines}) target_link_libraries(function_source qjs) # Examples # if(BUILD_EXAMPLES) add_executable(hello gen/hello.c ) add_qjs_libc_if_needed(hello) target_include_directories(hello PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) target_compile_definitions(hello PRIVATE ${qjs_defines}) target_link_libraries(hello qjs) add_executable(hello_module gen/hello_module.c ) add_qjs_libc_if_needed(hello_module) target_include_directories(hello_module PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) target_compile_definitions(hello_module PRIVATE ${qjs_defines}) target_link_libraries(hello_module qjs) add_library(fib MODULE examples/fib.c) set_target_properties(fib PROPERTIES PREFIX "" C_VISIBILITY_PRESET default ) target_compile_definitions(fib PRIVATE JS_SHARED_LIBRARY) if(WIN32) target_link_libraries(fib qjs) elseif(APPLE) target_link_options(fib PRIVATE -undefined dynamic_lookup) endif() add_library(point MODULE examples/point.c) set_target_properties(point PROPERTIES PREFIX "" C_VISIBILITY_PRESET default ) target_compile_definitions(point PRIVATE JS_SHARED_LIBRARY) if(WIN32) target_link_libraries(point qjs) elseif(APPLE) target_link_options(point PRIVATE -undefined dynamic_lookup) endif() add_executable(test_fib examples/fib.c gen/test_fib.c ) add_qjs_libc_if_needed(test_fib) target_include_directories(test_fib PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}) target_compile_definitions(test_fib PRIVATE ${qjs_defines}) target_link_libraries(test_fib qjs) endif() add_executable(test_conv tests/test_conv.c ) # Install target # if(NOT IOS) file(STRINGS quickjs.h quickjs_h REGEX QJS_VERSION) string(REGEX MATCHALL "([0-9])" QJS_VERSION "${quickjs_h}") list(GET QJS_VERSION 0 QJS_VERSION_MAJOR) list(GET QJS_VERSION 1 QJS_VERSION_MINOR) list(GET QJS_VERSION 2 QJS_VERSION_PATCH) set_target_properties(qjs PROPERTIES VERSION ${QJS_VERSION_MAJOR}.${QJS_VERSION_MINOR}.${QJS_VERSION_PATCH} SOVERSION ${QJS_VERSION_MAJOR} ) install(FILES quickjs.h DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}) if(BUILD_QJS_LIBC) install(FILES quickjs-libc.h DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}) endif() install(TARGETS qjs_exe RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}) install(TARGETS qjsc RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}) install(TARGETS qjs EXPORT qjsConfig RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR} LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR} ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}) install(EXPORT qjsConfig DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/quickjs) install(FILES LICENSE DESTINATION ${CMAKE_INSTALL_DOCDIR}) install(DIRECTORY examples DESTINATION ${CMAKE_INSTALL_DOCDIR}) endif()