package SPVM::Builder::CC;
use strict;
use warnings;
use Carp 'confess';
use Config;
use ExtUtils::CBuilder;
use File::Copy 'copy', 'move';
use File::Path 'mkpath';
use File::Find 'find';
use File::Basename 'dirname', 'basename';
use SPVM::Builder::Util;
use SPVM::Builder::Config;
use SPVM::Builder::CompileInfo;
use SPVM::Builder::ObjectFileInfo;
use SPVM::Builder::LinkInfo;
use SPVM::Builder::Resource;
sub global_before_compile {
my $self = shift;
if (@_) {
$self->{global_before_compile} = $_[0];
return $self;
}
else {
return $self->{global_before_compile};
}
}
sub build_dir {
my $self = shift;
if (@_) {
$self->{build_dir} = $_[0];
return $self;
}
else {
return $self->{build_dir};
}
}
sub force {
my $self = shift;
if (@_) {
$self->{force} = $_[0];
return $self;
}
else {
return $self->{force};
}
}
sub quiet {
my $self = shift;
if (@_) {
$self->{quiet} = $_[0];
return $self;
}
else {
return $self->{quiet};
}
}
sub at_runtime {
my $self = shift;
if (@_) {
$self->{at_runtime} = $_[0];
return $self;
}
else {
return $self->{at_runtime};
}
}
sub debug {
my $self = shift;
if (@_) {
$self->{debug} = $_[0];
return $self;
}
else {
return $self->{debug};
}
}
sub output_type {
my $self = shift;
if (@_) {
$self->{output_type} = $_[0];
return $self;
}
else {
return $self->{output_type};
}
}
sub new {
my $class = shift;
my $self = {@_};
if ($ENV{SPVM_CC_DEBUG}) {
$self->{debug} = 1;
}
if ($ENV{SPVM_CC_FORCE}) {
$self->{force} = 1;
}
return bless $self, $class;
}
sub build_at_runtime {
my ($self, $class_name, $options) = @_;
$options ||= {};
my $dl_func_list = $options->{dl_func_list};
my $module_file = $options->{module_file};
my $precompile_source = $options->{precompile_source};
my $category = $options->{category};
# Build directory
my $build_dir = $self->build_dir;
if (defined $build_dir) {
mkpath $build_dir;
}
else {
confess "The \"build_dir\" field must be defined to build a $category method at runtime. Perhaps the setting of the SPVM_BUILD_DIR environment variable is forgotten";
}
# Source directory
my $build_src_dir;
if ($category eq 'precompile') {
$build_src_dir = SPVM::Builder::Util::create_build_src_path($self->build_dir);
mkpath $build_src_dir;
my $force = $self->detect_force;
$self->build_precompile_class_source_file(
$class_name,
{
output_dir => $build_src_dir,
force => $force,
precompile_source => $precompile_source,
module_file => $module_file,
}
);
}
elsif ($category eq 'native') {
my $module_file = $options->{module_file};
$build_src_dir = SPVM::Builder::Util::remove_class_part_from_file($module_file, $class_name);
}
# Object directory
my $build_object_dir = SPVM::Builder::Util::create_build_object_path($self->build_dir);
mkpath $build_object_dir;
# Lib directory
my $build_lib_dir = SPVM::Builder::Util::create_build_lib_path($self->build_dir);
mkpath $build_lib_dir;
my $build_file = $self->build(
$class_name,
{
compile_input_dir => $build_src_dir,
compile_output_dir => $build_object_dir,
link_output_dir => $build_lib_dir,
category => $category,
module_file => $module_file,
dl_func_list => $dl_func_list,
}
);
return $build_file;
}
sub build_dist {
my ($self, $class_name, $options) = @_;
$options ||= {};
my $dl_func_list = $options->{dl_func_list};
my $module_file = $options->{module_file};
my $precompile_source = $options->{precompile_source};
my $category = $options->{category};
my $build_src_dir;
if ($category eq 'precompile') {
$build_src_dir = SPVM::Builder::Util::create_build_src_path($self->build_dir);
mkpath $build_src_dir;
my $force = $self->detect_force;
$self->build_precompile_class_source_file(
$class_name,
{
output_dir => $build_src_dir,
force => $force,
precompile_source => $precompile_source,
module_file => $module_file,
}
);
}
elsif ($category eq 'native') {
$build_src_dir = 'lib';
}
my $build_object_dir = SPVM::Builder::Util::create_build_object_path($self->build_dir);
mkpath $build_object_dir;
my $build_lib_dir = 'blib/lib';
$self->build(
$class_name,
{
compile_input_dir => $build_src_dir,
compile_output_dir => $build_object_dir,
link_output_dir => $build_lib_dir,
category => $category,
module_file => $module_file,
dl_func_list => $dl_func_list,
}
);
}
sub build {
my ($self, $class_name, $options) = @_;
$options ||= {};
my $dl_func_list = $options->{dl_func_list};
my $category = $options->{category};
# Module file
my $module_file = $options->{module_file};
unless (defined $module_file) {
my $config_file = SPVM::Builder::Util::get_config_file_from_class_name($class_name);
if ($config_file) {
$module_file = $config_file;
$module_file =~ s/\.config$/\.spvm/;
}
else {
confess "\"$module_file\" module is not loaded";
}
}
my $config;
if ($category eq 'native') {
$config = $self->create_native_config_from_module_file($module_file);
}
elsif ($category eq 'precompile') {
$config = $self->create_precompile_config($module_file);
}
# Compile source file and create object files
my $compile_options = {
input_dir => $options->{compile_input_dir},
output_dir => $options->{compile_output_dir},
config => $config,
category => $category,
};
my $object_files = $self->compile($class_name, $compile_options);
# Link object files and create dynamic library
my $link_options = {
output_dir => $options->{link_output_dir},
config => $config,
category => $category,
dl_func_list => $dl_func_list,
};
my $output_file = $self->link(
$class_name,
$object_files,
$link_options
);
return $output_file;
}
sub resource_src_dir_from_class_name {
my ($self, $class_name) = @_;
my $config_file = SPVM::Builder::Util::get_config_file_from_class_name($class_name);
my $config_rel_file = SPVM::Builder::Util::convert_class_name_to_rel_file($class_name, 'config');
my $resource_src_dir = $config_file;
$resource_src_dir =~ s|/\Q$config_rel_file\E$||;
return $resource_src_dir;
}
sub get_resource_object_dir_from_class_name {
my ($self, $class_name) = @_;
my $class_rel_dir = SPVM::Builder::Util::convert_class_name_to_rel_file($class_name);
my $resource_object_dir = SPVM::Builder::Util::create_build_object_path($self->build_dir, "$class_rel_dir.resource");
return $resource_object_dir;
}
sub create_native_config_from_module_file {
my ($self, $module_file) = @_;
my $config;
my $config_file = $module_file;
$config_file =~ s/\.spvm$/.config/;
# Config file
if (-f $config_file) {
$config = SPVM::Builder::Config->load_config($config_file);
}
else {
my $error = $self->_error_message_find_config($config_file);
confess $error;
}
return $config;
}
sub create_precompile_config {
my ($self) = @_;
my $config = SPVM::Builder::Config->new_gnu99(file_optional => 1);
return $config;
}
sub detect_force {
my ($self, $config) = @_;
my $force;
if (defined $self->force) {
$force = $self->force;
}
elsif (defined $config && defined $config->force) {
$force = $config->force;
}
else {
$force = 0;
}
return $force;
}
sub detect_quiet {
my ($self, $config) = @_;
my $quiet;
if (defined $self->debug) {
$quiet = 0;
}
elsif (defined $self->quiet) {
$quiet = $self->quiet;
}
elsif (defined $config && defined $config->quiet) {
$quiet = $config->quiet;
}
elsif ($self->at_runtime) {
$quiet = 1;
}
else {
$quiet = 0;
}
return $quiet;
}
sub compile_single {
my ($self, $compile_info, $config) = @_;
# Quiet output
my $quiet = $self->detect_quiet($config);
my $source_file = $compile_info->source_file;
# Execute compile command
my $cbuilder = ExtUtils::CBuilder->new(quiet => 1);
my $cc_cmd = $compile_info->create_compile_command;
$cbuilder->do_system(@$cc_cmd)
or confess "Can't compile $source_file: @$cc_cmd";
unless ($quiet) {
warn "@$cc_cmd\n";
}
}
sub compile {
my ($self, $class_name, $options) = @_;
$options ||= {};
# Category
my $category = $options->{category};
# Build directory
my $build_dir = $self->build_dir;
if (defined $build_dir) {
mkpath $build_dir;
}
else {
confess "Build directory is not specified. Maybe forget to set \"SPVM_BUILD_DIR\" environment variable?";
}
# Input directory
my $input_dir = $options->{input_dir};
# Object directory
my $output_dir = $options->{output_dir};
unless (defined $output_dir && -d $output_dir) {
confess "Output directory must exists for " . $options->{category} . " build";
}
# Config
my $config = $options->{config};
# Force compile
my $force = $self->detect_force($config);
my $no_use_resource = $options->{no_use_resource};
my $ignore_native_module = $options->{ignore_native_module};
# Native module file
my $native_module_file;
unless ($ignore_native_module) {
# Native module file
my $native_module_ext = $config->ext;
unless (defined $native_module_ext) {
confess "Source extension is not specified";
}
my $native_module_rel_file = SPVM::Builder::Util::convert_class_name_to_category_rel_file($class_name, $category, $native_module_ext);
$native_module_file = "$input_dir/$native_module_rel_file";
unless (-f $native_module_file) {
confess "Can't find source file $native_module_file";
}
}
# Own resource source files
my $own_source_files = $config->source_files;
my $own_src_dir = $config->own_src_dir;
my $resource_src_files;
if (defined $own_src_dir) {
$resource_src_files = [map { "$own_src_dir/$_" } @$own_source_files ];
}
# Compile source files
my $object_file_infos = [];
my $is_native_module = 1;
for my $source_file ($native_module_file, @$resource_src_files) {
my $cur_is_native_module = $is_native_module;
$is_native_module = 0;
next unless defined $source_file;
my $object_file;
# Object file of native module
if ($cur_is_native_module) {
my $object_rel_file = SPVM::Builder::Util::convert_class_name_to_category_rel_file($class_name, $category, 'o');
$object_file = "$output_dir/$object_rel_file";
}
# Object file of resource source file
else {
my $object_rel_file = SPVM::Builder::Util::convert_class_name_to_category_rel_file($class_name, $category, 'native');
my $object_file_base = $source_file;
$object_file_base =~ s/^\Q$own_src_dir//;
$object_file_base =~ s/^[\\\/]//;
$object_file_base =~ s/\.[^\.]+$/.o/;
$object_file = "$output_dir/$object_rel_file/$object_file_base";
my $output_dir = dirname $object_file;
mkpath $output_dir;
}
# Check if object file need to be generated
my $need_generate;
{
# Own resource header files
my @own_header_files;
my $own_include_dir = $config->own_include_dir;
if (defined $own_include_dir && -d $own_include_dir) {
find(
{
wanted => sub {
my $include_file_name = $File::Find::name;
if (-f $include_file_name) {
push @own_header_files, $include_file_name;
}
},
no_chdir => 1,
},
$own_include_dir,
);
}
my $input_files = [$source_file, @own_header_files];
if (defined $config->file) {
push @$input_files, $config->file;
};
if ($cur_is_native_module) {
my $module_file = $source_file;
$module_file =~ s/\.[^\/\\]+$//;
$module_file .= '.spvm';
push @$input_files, $module_file;
}
$need_generate = SPVM::Builder::Util::need_generate({
force => $force,
output_file => $object_file,
input_files => $input_files,
});
}
# Compile-information
my $compile_info = $self->create_compile_command_info({
class_name => $class_name,
config => $config,
output_file => $object_file,
source_file => $source_file,
include_dirs => $options->{include_dirs},
no_use_resource => $no_use_resource,
});
if (defined $config->before_compile) {
$config->before_compile->($config, $compile_info);
}
if (defined $self->global_before_compile) {
$self->global_before_compile->($config, $compile_info);
}
# Compile a source file
if ($need_generate) {
my $class_rel_dir = SPVM::Builder::Util::convert_class_name_to_rel_dir($class_name);
my $work_output_dir = "$output_dir/$class_rel_dir";
mkpath dirname $object_file;
$self->compile_single($compile_info, $config);
}
# Object file information
my $compile_info_cc = $compile_info->{cc};
my $compile_info_ccflags = $compile_info->{ccflags};
my $object_file_info = SPVM::Builder::ObjectFileInfo->new(
compile_info => $compile_info,
);
# Add object file information
push @$object_file_infos, $object_file_info;
}
return $object_file_infos;
}
sub create_compile_command {
my ($self, $compile_info) = @_;
my $cc = $compile_info->{cc};
my $merged_ccflags = $compile_info->create_merged_ccflags;
my $object_file = $compile_info->{object_file};
my $source_file = $compile_info->{source_file};
my $cc_cmd = [$cc, '-c', @$merged_ccflags, '-o', $object_file, $source_file];
return $cc_cmd;
}
sub create_compile_command_info {
my ($self, $options) = @_;
unless ($options) {
$options = {};
}
my $class_name = $options->{class_name};
my $config = $options->{config};
my $output_file = $options->{output_file};
my $source_file = $options->{source_file};
my $cc = $config->cc;
# Include directories
my $no_use_resource = $options->{no_use_resource};
my @include_dirs = @{$config->include_dirs};
{
# Add own resource include directory
my $own_include_dir = $config->own_include_dir;
if (defined $own_include_dir) {
push @include_dirs, $own_include_dir;
}
# Add resource include directories
unless ($options->{no_use_resource}) {
my $resource_names = $config->get_resource_names;
for my $resource_name (@$resource_names) {
my $resource = $config->get_resource($resource_name);
my $config = $resource->config;
my $resource_include_dir = $config->own_include_dir;
if (defined $resource_include_dir) {
push @include_dirs, $resource_include_dir;
}
}
}
# Add option include directories
if (defined $options->{include_dirs}) {
push @include_dirs, @{$options->{include_dirs}};
}
}
my $ccflags = $config->ccflags;
my $optimize = $config->optimize;
my $builder_include_dir = $config->builder_include_dir;
my $compile_info = SPVM::Builder::CompileInfo->new(
cc => $cc,
ccflags => $ccflags,
output_file => $output_file,
source_file => $source_file,
optimize => $optimize,
builder_include_dir => $builder_include_dir,
include_dirs => \@include_dirs,
config => $config,
);
return $compile_info;
}
sub _error_message_find_config {
my ($self, $config_file) = @_;
my $error = <<"EOS";
Can't find the native config file \"$config_file\".
The config file must contain at least the following code.
----------------------------------------------
use strict;
use warnings;
use SPVM::Builder::Config;
my \$config = SPVM::Builder::Config->new_gnu99(file => __FILE__);
\$config;
----------------------------------------------
EOS
}
sub link {
my ($self, $class_name, $object_file_infos, $options) = @_;
my $dl_func_list = $options->{dl_func_list};
my $category = $options->{category};
# Build directory
my $build_dir = $self->build_dir;
if (defined $build_dir) {
mkpath $build_dir;
}
else {
confess "The \"build_dir\" field must be defined to build the native module for the $category methods. Perhaps the setting of the SPVM_BUILD_DIR environment variable is forgotten";
}
# Config
my $config = $options->{config};
unless ($config) {
confess "Need config option";
}
# Force link
my $force = $self->detect_force($config);
# Link information
my $link_info = $self->create_link_info($class_name, $object_file_infos, $config, $options);
# Output file
my $output_file = $link_info->output_file;
# Execute the callback before this link
my $before_link = $config->before_link;
if ($before_link) {
$before_link->($config, $link_info);
}
my @object_files = map { "$_" } @{$link_info->object_file_infos};
my $input_files = [@object_files];
if (defined $config->file) {
push @$input_files, $config->file;
}
my $need_generate = SPVM::Builder::Util::need_generate({
force => $force,
output_file => $output_file,
input_files => $input_files,
});
if ($need_generate) {
# Move temporary dynamic library file to blib directory
mkpath dirname $output_file;
my $ld = $link_info->ld;
my $cbuilder_config = {
ld => $ld,
lddlflags => '',
shrpenv => '',
libpth => '',
libperl => '',
# "perllibs" should be empty string, but ExtUtils::CBuiler outputs "INPUT()" into
# Linker Script File(.lds) when "perllibs" is empty string.
# This is syntax error in Linker Script File(.lds)
# For the reason, libm is linked which seems to have no effect.
perllibs => '-lm',
};
# Quiet output
my $quiet = $self->detect_quiet($config);
# ExtUtils::CBuilder object
my $cbuilder = ExtUtils::CBuilder->new(quiet => 1, config => $cbuilder_config);
my $link_info_ld = $link_info->ld;
my $link_info_class_name = $link_info->class_name;
my $link_info_output_file = $link_info->output_file;
my $link_info_object_file_infos = $link_info->object_file_infos;
my $merged_ldflags = $link_info->create_merged_ldflags;
my $link_info_object_files = [map { my $tmp = $_->to_string; $tmp } @$link_info_object_file_infos];
my @tmp_files;
my $output_type = $config->output_type;
# Create a dynamic library
if ($output_type eq 'dynamic_lib') {
(undef, @tmp_files) = $cbuilder->link(
objects => $link_info_object_files,
module_name => $link_info_class_name,
lib_file => $link_info_output_file,
extra_linker_flags => "@$merged_ldflags",
dl_func_list => $dl_func_list,
);
unless ($quiet) {
my $link_command = $link_info->to_string;
warn "$link_command\n";
}
}
# Create a static library
elsif ($output_type eq 'static_lib') {
my @object_files = map { "$_" } @$link_info_object_files;
my @ar_cmd = ('ar', 'rc', $link_info_output_file, @object_files);
$cbuilder->do_system(@ar_cmd)
or confess "Can't execute command @ar_cmd";
unless ($quiet) {
warn "@ar_cmd\n";
}
}
# Create an executable file
elsif ($output_type eq 'exe') {
(undef, @tmp_files) = $cbuilder->link_executable(
objects => $link_info_object_files,
module_name => $link_info_class_name,
exe_file => $link_info_output_file,
extra_linker_flags => "@$merged_ldflags",
);
unless ($quiet) {
my $link_command = $link_info->to_string;
warn "$link_command\n";
}
}
else {
confess "Unknown output_type \"$output_type\"";
}
if ($self->debug) {
if ($^O eq 'MSWin32') {
my $def_file;
my $lds_file;
for my $tmp_file (@tmp_files) {
# Remove double quote
$tmp_file =~ s/^"//;
$tmp_file =~ s/"$//;
if ($tmp_file =~ /\.def$/) {
$def_file = $tmp_file;
$lds_file = $def_file;
$lds_file =~ s/\.def$/.lds/;
last;
}
}
if (defined $def_file && -f $def_file) {
my $def_content = SPVM::Builder::Util::slurp_binary($def_file);
warn "[$def_file]\n$def_content\n";
}
if (defined $lds_file && -f $lds_file) {
my $lds_content = SPVM::Builder::Util::slurp_binary($lds_file);
warn "[$lds_file]\n$lds_content\n";
}
}
}
}
return $output_file;
}
sub create_link_info {
my ($self, $class_name, $object_file_infos, $config, $options) = @_;
my $category = $options->{category};
my $all_object_file_infos = [@$object_file_infos];
$options ||= {};
# Linker
my $ld = $config->ld;
# Output type
my $output_type = $self->output_type || $config->output_type;
# Libraries
my $lib_infos = [];
my $libs = $config->libs;
my $lib_dirs = $config->lib_dirs;
for my $lib (@$libs) {
my $lib_info;
# Library is linked by file path
my $static;
my $lib_name;
my $file_flag;
if (ref $lib) {
$static = $lib->static;
$lib_name = $lib->name;
$file_flag = $lib->file_flag;
$lib_info = $lib;
}
else {
$lib_name = $lib;
$lib_info = SPVM::Builder::LibInfo->new;
$lib_info->name($lib_name);
}
$lib_info->config($config);
if ($file_flag) {
my $found_lib_file;
for my $lib_dir (@$lib_dirs) {
$lib_dir =~ s|[\\/]$||;
# Search dynamic library
unless ($static) {
my $dynamic_lib_file_base = "lib$lib_name.$Config{dlext}";
my $dynamic_lib_file = "$lib_dir/$dynamic_lib_file_base";
if (-f $dynamic_lib_file) {
$found_lib_file = $dynamic_lib_file;
last;
}
}
# Search static library
my $static_lib_file_base = "lib$lib_name.a";
my $static_lib_file = "$lib_dir/$static_lib_file_base";
if (-f $static_lib_file) {
$found_lib_file = $static_lib_file;
last;
}
}
if (defined $found_lib_file) {
$lib_info->file = $found_lib_file;
}
}
push @$lib_infos, $lib_info;
}
# Use resources
my $resource_names = $config->get_resource_names;
my $resource_include_dirs = [];
for my $resource_name (@$resource_names) {
my $resource = $config->get_resource($resource_name);
my $resource_config = $resource->config;
my $resource_include_dir = $resource_config->own_include_dir;
if (defined $resource_include_dir) {
push @$resource_include_dirs, $resource_include_dir;
}
}
for my $resource_name (@$resource_names) {
my $resource = $config->get_resource($resource_name);
# Build native classes
my $builder_cc_resource = SPVM::Builder::CC->new(
build_dir => $self->build_dir,
);
my $resource_src_dir = $self->resource_src_dir_from_class_name($resource);
my $resource_object_dir = $self->get_resource_object_dir_from_class_name($class_name);
mkpath $resource_object_dir;
my $resource_class_name;
my $resource_config;
if (ref $resource) {
$resource_class_name = $resource->class_name;
$resource_config = $resource->config;
}
else {
$resource_class_name = $resource;
}
my $compile_options = {
input_dir => $resource_src_dir,
output_dir => $resource_object_dir,
no_use_resource => 1,
ignore_native_module => 1,
config => $resource_config,
category => $category,
};
if ($resource_config) {
$compile_options->{config} = $resource_config;
}
$compile_options->{include_dirs} = $resource_include_dirs;
my $object_file_infos = $builder_cc_resource->compile($resource_class_name, $compile_options);
push @$all_object_file_infos, @$object_file_infos;
}
my $all_object_files = [map { $_->to_string } @$all_object_file_infos];
# Output file
my $output_file = $options->{output_file};
unless (defined $output_file) {
# Dynamic library directory
my $output_dir = $options->{output_dir};
unless (defined $output_dir && -d $output_dir) {
confess "Shared lib directory must be specified for link";
}
# Dynamic library file
my $output_rel_file = SPVM::Builder::Util::convert_class_name_to_category_rel_file($class_name, $options->{category});
$output_file = "$output_dir/$output_rel_file";
}
# Add output file extension
my $output_file_base = basename $output_file;
if ($output_file_base =~ /\.precompile$/ || $output_file_base !~ /\./) {
my $exe_ext;
# Dynamic library
if ($output_type eq 'dynamic_lib') {
$exe_ext = ".$Config{dlext}"
}
# Static library
elsif ($output_type eq 'static_lib') {
$exe_ext = '.a';
}
# Executable file
elsif ($output_type eq 'exe') {
$exe_ext = $Config{exe_ext};
}
$output_file .= $exe_ext;
}
# Linker flags for dynamic link
my $dynamic_lib_ldflags = $config->dynamic_lib_ldflags;
# Linker flags
my $ldflags = $config->ldflags;
# Optimize
my $ld_optimize = $config->ld_optimize;
my $link_info = SPVM::Builder::LinkInfo->new(
class_name => $class_name,
ld => $ld,
ldflags => $ldflags,
lib_infos => $lib_infos,
object_file_infos => $all_object_file_infos,
output_file => $output_file,
lib_dirs => $lib_dirs,
ld_optimize => $ld_optimize,
dynamic_lib_ldflags => $dynamic_lib_ldflags,
output_type => $output_type,
);
return $link_info;
}
sub build_precompile_class_source_file {
my ($self, $class_name, $options) = @_;
my $precompile_source = $options->{precompile_source};
my $module_file = $options->{module_file};
# Force
my $force = $options->{force};
# Output - Precompile C source file
my $output_dir = $options->{output_dir};
my $source_rel_file = SPVM::Builder::Util::convert_class_name_to_rel_file($class_name, 'precompile.c');
my $source_file = "$output_dir/$source_rel_file";
# Check if generating is needed
my $spvm_module_dir = $INC{'SPVM/Builder.pm'};
$spvm_module_dir =~ s/\.pm$//;
$spvm_module_dir .= '/src';
my $spvm_precompile_soruce_file = "$spvm_module_dir/spvm_precompile.c";
unless (-f $spvm_precompile_soruce_file) {
confess "Can't find $spvm_precompile_soruce_file";
}
my $need_generate = SPVM::Builder::Util::need_generate({
force => $force,
output_file => $source_file,
input_files => [$module_file, $spvm_precompile_soruce_file],
});
# Generate precompile C source file
if ($need_generate) {
mkpath dirname $source_file;
open my $fh, '>', $source_file
or die "Can't create $source_file";
print $fh $precompile_source;
close $fh;
}
}
1;
=head1 Name
SPVM::Builder::CC - Compiler and Linker of Native Sources
=head1 Copyright & License
Copyright (c) 2023 Yuki Kimoto
MIT License