2023-03-18 17:32:21 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
2024-01-10 21:08:25 +08:00
|
|
|
namespace SPC\builder\unix;
|
2023-03-18 17:32:21 +08:00
|
|
|
|
2024-01-10 21:08:25 +08:00
|
|
|
use SPC\builder\BuilderBase;
|
2023-04-29 18:59:47 +08:00
|
|
|
use SPC\builder\linux\LinuxBuilder;
|
2023-03-26 22:27:51 +08:00
|
|
|
use SPC\exception\FileSystemException;
|
2023-03-18 17:32:21 +08:00
|
|
|
use SPC\exception\RuntimeException;
|
2023-08-20 19:51:45 +08:00
|
|
|
use SPC\exception\WrongUsageException;
|
2024-01-10 21:08:25 +08:00
|
|
|
use SPC\store\Config;
|
2023-03-26 22:27:51 +08:00
|
|
|
use SPC\store\FileSystem;
|
2024-01-10 21:08:25 +08:00
|
|
|
use SPC\util\DependencyUtil;
|
2024-12-10 23:08:01 +08:00
|
|
|
use SPC\util\SPCConfigUtil;
|
2023-03-18 17:32:21 +08:00
|
|
|
|
2024-01-10 21:08:25 +08:00
|
|
|
abstract class UnixBuilderBase extends BuilderBase
|
2023-03-18 17:32:21 +08:00
|
|
|
{
|
2023-08-20 19:51:45 +08:00
|
|
|
/** @var string cflags */
|
2023-03-18 17:32:21 +08:00
|
|
|
public string $arch_c_flags;
|
|
|
|
|
|
2023-08-20 19:51:45 +08:00
|
|
|
/** @var string C++ flags */
|
2023-03-18 17:32:21 +08:00
|
|
|
public string $arch_cxx_flags;
|
|
|
|
|
|
|
|
|
|
/** @var string cmake toolchain file */
|
|
|
|
|
public string $cmake_toolchain_file;
|
|
|
|
|
|
2023-08-20 19:51:45 +08:00
|
|
|
/**
|
|
|
|
|
* @throws WrongUsageException
|
|
|
|
|
* @throws FileSystemException
|
|
|
|
|
*/
|
2023-03-18 17:32:21 +08:00
|
|
|
public function getAllStaticLibFiles(): array
|
|
|
|
|
{
|
|
|
|
|
$libs = [];
|
|
|
|
|
|
|
|
|
|
// reorder libs
|
|
|
|
|
foreach ($this->libs as $lib) {
|
|
|
|
|
foreach ($lib->getDependencies() as $dep) {
|
|
|
|
|
$libs[] = $dep;
|
|
|
|
|
}
|
|
|
|
|
$libs[] = $lib;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$libFiles = [];
|
|
|
|
|
$libNames = [];
|
|
|
|
|
// merge libs
|
|
|
|
|
foreach ($libs as $lib) {
|
|
|
|
|
if (!in_array($lib::NAME, $libNames, true)) {
|
|
|
|
|
$libNames[] = $lib::NAME;
|
|
|
|
|
array_unshift($libFiles, ...$lib->getStaticLibs());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return array_map(fn ($x) => realpath(BUILD_LIB_PATH . "/{$x}"), $libFiles);
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-10 11:10:40 +08:00
|
|
|
/**
|
|
|
|
|
* Return generic cmake options when configuring cmake projects
|
|
|
|
|
*/
|
|
|
|
|
public function makeCmakeArgs(): string
|
|
|
|
|
{
|
|
|
|
|
$extra = $this instanceof LinuxBuilder ? '-DCMAKE_C_COMPILER=' . getenv('CC') . ' ' : '';
|
|
|
|
|
return $extra .
|
|
|
|
|
'-DCMAKE_BUILD_TYPE=Release ' .
|
|
|
|
|
'-DCMAKE_INSTALL_PREFIX=/ ' .
|
|
|
|
|
'-DCMAKE_INSTALL_BINDIR=/bin ' .
|
|
|
|
|
'-DCMAKE_INSTALL_LIBDIR=/lib ' .
|
|
|
|
|
'-DCMAKE_INSTALL_INCLUDEDIR=/include ' .
|
|
|
|
|
"-DCMAKE_TOOLCHAIN_FILE={$this->cmake_toolchain_file}";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Generate configure flags
|
|
|
|
|
*/
|
|
|
|
|
public function makeAutoconfFlags(int $flag = AUTOCONF_ALL): string
|
|
|
|
|
{
|
|
|
|
|
$extra = '';
|
|
|
|
|
// TODO: add auto pkg-config support
|
|
|
|
|
if (($flag & AUTOCONF_LIBS) === AUTOCONF_LIBS) {
|
|
|
|
|
$extra .= 'LIBS="' . BUILD_LIB_PATH . '" ';
|
|
|
|
|
}
|
|
|
|
|
if (($flag & AUTOCONF_CFLAGS) === AUTOCONF_CFLAGS) {
|
|
|
|
|
$extra .= 'CFLAGS="-I' . BUILD_INCLUDE_PATH . '" ';
|
|
|
|
|
}
|
|
|
|
|
if (($flag & AUTOCONF_CPPFLAGS) === AUTOCONF_CPPFLAGS) {
|
|
|
|
|
$extra .= 'CPPFLAGS="-I' . BUILD_INCLUDE_PATH . '" ';
|
|
|
|
|
}
|
|
|
|
|
if (($flag & AUTOCONF_LDFLAGS) === AUTOCONF_LDFLAGS) {
|
|
|
|
|
$extra .= 'LDFLAGS="-L' . BUILD_LIB_PATH . '" ';
|
|
|
|
|
}
|
|
|
|
|
return $extra;
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-16 13:01:11 +08:00
|
|
|
public function proveLibs(array $sorted_libraries): void
|
2024-01-10 21:08:25 +08:00
|
|
|
{
|
|
|
|
|
// search all supported libs
|
|
|
|
|
$support_lib_list = [];
|
|
|
|
|
$classes = FileSystem::getClassesPsr4(
|
|
|
|
|
ROOT_DIR . '/src/SPC/builder/' . osfamily2dir() . '/library',
|
2024-06-20 14:46:08 +08:00
|
|
|
'SPC\builder\\' . osfamily2dir() . '\library'
|
2024-01-10 21:08:25 +08:00
|
|
|
);
|
|
|
|
|
foreach ($classes as $class) {
|
|
|
|
|
if (defined($class . '::NAME') && $class::NAME !== 'unknown' && Config::getLib($class::NAME) !== null) {
|
|
|
|
|
$support_lib_list[$class::NAME] = $class;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// if no libs specified, compile all supported libs
|
|
|
|
|
if ($sorted_libraries === [] && $this->isLibsOnly()) {
|
|
|
|
|
$libraries = array_keys($support_lib_list);
|
2024-02-16 18:56:33 +08:00
|
|
|
$sorted_libraries = DependencyUtil::getLibs($libraries);
|
2024-01-10 21:08:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// add lib object for builder
|
|
|
|
|
foreach ($sorted_libraries as $library) {
|
2025-03-08 14:29:44 +08:00
|
|
|
if (!in_array(Config::getLib($library, 'type', 'lib'), ['lib', 'package'])) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2024-01-10 21:08:25 +08:00
|
|
|
// if some libs are not supported (but in config "lib.json", throw exception)
|
|
|
|
|
if (!isset($support_lib_list[$library])) {
|
|
|
|
|
throw new WrongUsageException('library [' . $library . '] is in the lib.json list but not supported to compile, but in the future I will support it!');
|
|
|
|
|
}
|
|
|
|
|
$lib = new ($support_lib_list[$library])($this);
|
|
|
|
|
$this->addLib($lib);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// calculate and check dependencies
|
|
|
|
|
foreach ($this->libs as $lib) {
|
|
|
|
|
$lib->calcDependency();
|
|
|
|
|
}
|
2024-12-10 23:08:01 +08:00
|
|
|
$this->lib_list = $sorted_libraries;
|
2024-01-10 21:08:25 +08:00
|
|
|
}
|
|
|
|
|
|
2023-03-18 17:32:21 +08:00
|
|
|
/**
|
2023-04-23 20:31:58 +08:00
|
|
|
* Sanity check after build complete
|
|
|
|
|
*
|
2023-03-18 17:32:21 +08:00
|
|
|
* @throws RuntimeException
|
|
|
|
|
*/
|
2024-01-10 11:10:40 +08:00
|
|
|
protected function sanityCheck(int $build_target): void
|
2023-03-18 17:32:21 +08:00
|
|
|
{
|
2023-04-23 20:31:58 +08:00
|
|
|
// sanity check for php-cli
|
|
|
|
|
if (($build_target & BUILD_TARGET_CLI) === BUILD_TARGET_CLI) {
|
|
|
|
|
logger()->info('running cli sanity check');
|
2025-02-16 02:30:08 +09:00
|
|
|
[$ret, $output] = shell()->execWithResult(BUILD_ROOT_PATH . '/bin/php -n -r "echo \"hello\";"');
|
2025-02-11 03:34:43 +01:00
|
|
|
$raw_output = implode('', $output);
|
|
|
|
|
if ($ret !== 0 || trim($raw_output) !== 'hello') {
|
|
|
|
|
throw new RuntimeException("cli failed sanity check: ret[{$ret}]. out[{$raw_output}]");
|
2023-03-18 17:32:21 +08:00
|
|
|
}
|
2023-09-13 13:07:22 +02:00
|
|
|
|
2023-03-18 17:32:21 +08:00
|
|
|
foreach ($this->exts as $ext) {
|
2023-03-26 22:27:51 +08:00
|
|
|
logger()->debug('testing ext: ' . $ext->getName());
|
2024-01-10 21:08:25 +08:00
|
|
|
$ext->runCliCheckUnix();
|
2023-03-18 17:32:21 +08:00
|
|
|
}
|
|
|
|
|
}
|
2023-04-23 20:31:58 +08:00
|
|
|
|
|
|
|
|
// sanity check for phpmicro
|
|
|
|
|
if (($build_target & BUILD_TARGET_MICRO) === BUILD_TARGET_MICRO) {
|
2024-06-03 23:16:15 +08:00
|
|
|
$test_task = $this->getMicroTestTasks();
|
|
|
|
|
foreach ($test_task as $task_name => $task) {
|
|
|
|
|
$test_file = SOURCE_PATH . '/' . $task_name . '.exe';
|
|
|
|
|
if (file_exists($test_file)) {
|
|
|
|
|
@unlink($test_file);
|
|
|
|
|
}
|
|
|
|
|
file_put_contents($test_file, file_get_contents(SOURCE_PATH . '/php-src/sapi/micro/micro.sfx') . $task['content']);
|
|
|
|
|
chmod($test_file, 0755);
|
|
|
|
|
[$ret, $out] = shell()->execWithResult($test_file);
|
|
|
|
|
foreach ($task['conditions'] as $condition => $closure) {
|
|
|
|
|
if (!$closure($ret, $out)) {
|
|
|
|
|
$raw_out = trim(implode('', $out));
|
|
|
|
|
throw new RuntimeException("micro failed sanity check: {$task_name}, condition [{$condition}], ret[{$ret}], out[{$raw_out}]");
|
|
|
|
|
}
|
2024-01-29 10:04:21 +08:00
|
|
|
}
|
2023-03-18 17:32:21 +08:00
|
|
|
}
|
|
|
|
|
}
|
2024-12-10 23:08:01 +08:00
|
|
|
|
|
|
|
|
// sanity check for embed
|
|
|
|
|
if (($build_target & BUILD_TARGET_EMBED) === BUILD_TARGET_EMBED) {
|
|
|
|
|
logger()->info('running embed sanity check');
|
|
|
|
|
$sample_file_path = SOURCE_PATH . '/embed-test';
|
|
|
|
|
if (!is_dir($sample_file_path)) {
|
|
|
|
|
@mkdir($sample_file_path);
|
|
|
|
|
}
|
|
|
|
|
// copy embed test files
|
|
|
|
|
copy(ROOT_DIR . '/src/globals/common-tests/embed.c', $sample_file_path . '/embed.c');
|
|
|
|
|
copy(ROOT_DIR . '/src/globals/common-tests/embed.php', $sample_file_path . '/embed.php');
|
|
|
|
|
$util = new SPCConfigUtil($this);
|
|
|
|
|
$config = $util->config($this->ext_list, $this->lib_list, $this->getOption('with-suggested-exts'), $this->getOption('with-suggested-libs'));
|
|
|
|
|
$lens = "{$config['cflags']} {$config['ldflags']} {$config['libs']}";
|
2025-02-04 17:34:47 +08:00
|
|
|
if (PHP_OS_FAMILY === 'Linux' && $this->getOption('libc') !== 'glibc') {
|
2024-12-10 23:08:01 +08:00
|
|
|
$lens .= ' -static';
|
|
|
|
|
}
|
|
|
|
|
[$ret, $out] = shell()->cd($sample_file_path)->execWithResult(getenv('CC') . ' -o embed embed.c ' . $lens);
|
|
|
|
|
if ($ret !== 0) {
|
|
|
|
|
throw new RuntimeException('embed failed sanity check: build failed. Error message: ' . implode("\n", $out));
|
|
|
|
|
}
|
|
|
|
|
[$ret, $output] = shell()->cd($sample_file_path)->execWithResult('./embed');
|
|
|
|
|
if ($ret !== 0 || trim(implode('', $output)) !== 'hello') {
|
|
|
|
|
throw new RuntimeException('embed failed sanity check: run failed. Error message: ' . implode("\n", $output));
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-03-18 17:32:21 +08:00
|
|
|
}
|
2023-03-26 22:27:51 +08:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 将编译好的二进制文件发布到 buildroot
|
|
|
|
|
*
|
|
|
|
|
* @param int $type 发布类型
|
|
|
|
|
* @throws RuntimeException
|
|
|
|
|
* @throws FileSystemException
|
|
|
|
|
*/
|
2024-01-10 11:10:40 +08:00
|
|
|
protected function deployBinary(int $type): bool
|
2023-03-26 22:27:51 +08:00
|
|
|
{
|
|
|
|
|
$src = match ($type) {
|
2023-04-23 20:31:58 +08:00
|
|
|
BUILD_TARGET_CLI => SOURCE_PATH . '/php-src/sapi/cli/php',
|
|
|
|
|
BUILD_TARGET_MICRO => SOURCE_PATH . '/php-src/sapi/micro/micro.sfx',
|
|
|
|
|
BUILD_TARGET_FPM => SOURCE_PATH . '/php-src/sapi/fpm/php-fpm',
|
2023-03-26 22:27:51 +08:00
|
|
|
default => throw new RuntimeException('Deployment does not accept type ' . $type),
|
|
|
|
|
};
|
2023-04-23 20:31:58 +08:00
|
|
|
logger()->info('Deploying ' . $this->getBuildTypeName($type) . ' file');
|
2023-03-26 22:27:51 +08:00
|
|
|
FileSystem::createDir(BUILD_ROOT_PATH . '/bin');
|
|
|
|
|
shell()->exec('cp ' . escapeshellarg($src) . ' ' . escapeshellarg(BUILD_ROOT_PATH . '/bin/'));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2023-08-20 19:51:45 +08:00
|
|
|
* Run php clean
|
2023-03-26 22:27:51 +08:00
|
|
|
*
|
|
|
|
|
* @throws RuntimeException
|
|
|
|
|
*/
|
2024-01-10 11:10:40 +08:00
|
|
|
protected function cleanMake(): void
|
2023-03-26 22:27:51 +08:00
|
|
|
{
|
|
|
|
|
logger()->info('cleaning up');
|
|
|
|
|
shell()->cd(SOURCE_PATH . '/php-src')->exec('make clean');
|
|
|
|
|
}
|
2023-03-18 17:32:21 +08:00
|
|
|
}
|