Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: init shared library support. #1050

Open
wants to merge 5 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 10 additions & 2 deletions src/fpm.f90
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,14 @@ module fpm
use fpm_model, only: fpm_model_t, srcfile_t, show_model, fortran_features_t, &
FPM_SCOPE_UNKNOWN, FPM_SCOPE_LIB, FPM_SCOPE_DEP, &
FPM_SCOPE_APP, FPM_SCOPE_EXAMPLE, FPM_SCOPE_TEST
use fpm_compiler, only: new_compiler, new_archiver, set_cpp_preprocessor_flags
use fpm_compiler, only: new_compiler, new_archiver, set_cpp_preprocessor_flags, &
generate_shared_library


use fpm_sources, only: add_executable_sources, add_sources_from_dir
use fpm_targets, only: targets_from_sources, build_target_t, build_target_ptr, &
FPM_TARGET_EXECUTABLE, FPM_TARGET_ARCHIVE
FPM_TARGET_EXECUTABLE, FPM_TARGET_ARCHIVE, &
FPM_TARGET_OBJECT
use fpm_manifest, only : get_package_data, package_config_t
use fpm_meta, only : resolve_metapackages
use fpm_error, only : error_t, fatal_error, fpm_stop
Expand Down Expand Up @@ -467,6 +469,12 @@ subroutine cmd_build(settings)
call build_package(targets,model,verbose=settings%verbose)
endif

do i=1, size(targets)
if (targets(i)%ptr%target_type == FPM_TARGET_ARCHIVE .and. package%build%shared_library) then
call model%compiler%generate_shared_library(model%package_name, targets(i)%ptr%output_file)
exit
end if
enddo
end subroutine cmd_build

subroutine cmd_run(settings,test)
Expand Down
14 changes: 13 additions & 1 deletion src/fpm/manifest/build.f90
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,9 @@ module fpm_manifest_build
!> External modules to use
type(string_t), allocatable :: external_modules(:)

!> Generate a shared library.
logical :: shared_library = .false.

contains

!> Print information on this instance
Expand Down Expand Up @@ -135,6 +138,8 @@ subroutine new_build_config(self, table, package_name, error)
call get_list(table, "external-modules", self%external_modules, error)
if (allocated(error)) return

call get_value(table, "shared-library", self%shared_library, .false., stat=stat)

end subroutine new_build_config

!> Check local schema for allowed entries
Expand All @@ -160,7 +165,7 @@ subroutine check(table, package_name, error)
do ikey = 1, size(list)
select case(list(ikey)%key)

case("auto-executables", "auto-examples", "auto-tests", "link", "external-modules", "module-naming")
case("auto-executables", "auto-examples", "auto-tests", "link", "external-modules", "module-naming", "shared-library")
continue

case default
Expand Down Expand Up @@ -216,6 +221,8 @@ subroutine info(self, unit, verbosity)
end do
end if

write(unit, fmt) " - generate shared library ", merge("enabled ", "disabled", self%shared_library)

end subroutine info

!> Check that two dependency trees are equal
Expand All @@ -235,6 +242,7 @@ logical function build_conf_is_same(this,that)
if (.not.this%module_prefix==other%module_prefix) return
if (.not.this%link==other%link) return
if (.not.this%external_modules==other%external_modules) return
if (this%shared_library.neqv.other%shared_library) return

class default
! Not the same type
Expand Down Expand Up @@ -278,6 +286,8 @@ subroutine dump_to_toml(self, table, error)
call set_list(table, "external-modules", self%external_modules, error)
if (allocated(error)) return

call set_value(table, "shared-library", self%shared_library, error, class_name)

end subroutine dump_to_toml

!> Read build config from toml table (no checks made at this stage)
Expand Down Expand Up @@ -321,6 +331,8 @@ subroutine load_from_toml(self, table, error)
call get_list(table, "external-modules", self%external_modules, error)
if (allocated(error)) return

call get_value(table, "shared-library", self%shared_library, error, class_name)

end subroutine load_from_toml


Expand Down
1 change: 1 addition & 0 deletions src/fpm/manifest/install.f90
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@ logical function install_conf_same(this,that)

install_conf_same = .false.


select type (other=>that)
type is (install_config_t)
if (this%library.neqv.other%library) return
Expand Down
15 changes: 15 additions & 0 deletions src/fpm_compiler.F90
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ module fpm_compiler
implicit none
public :: compiler_t, new_compiler, archiver_t, new_archiver, get_macros
public :: debug
public :: generate_shared_library

enum, bind(C)
enumerator :: &
Expand Down Expand Up @@ -114,6 +115,8 @@ module fpm_compiler
procedure :: is_gnu
!> Enumerate libraries, based on compiler and platform
procedure :: enumerate_libraries
!> Function for generating shared library.
procedure :: generate_shared_library

!> Serialization interface
procedure :: serializable_is_same => compiler_is_same
Expand Down Expand Up @@ -1104,6 +1107,18 @@ subroutine compile_fortran(self, input, output, args, log_file, stat)
& echo=self%echo, verbose=self%verbose, redirect=log_file, exitstat=stat)
end subroutine compile_fortran

subroutine generate_shared_library(self, package_name, output_file)
!> Instance of the compiler object
class(compiler_t), intent(in) :: self
!> Name of the package.
character(len=*), intent(in) :: package_name
!> Output file of library archive.
character(len=*), intent(in) :: output_file

call run(self%fc // " --shared " // " -o " // "lib" // package_name // ".so" // " " &
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would this command work for all compilers in the fpm suite, or there are compiler specific flags that should be used?

Also the .so output looks a bit odd to me: isn't a dynamic library supposed to be named .dll on Windows and .dylib on macOS?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the review @perazz . .so works well with Unix based operating systems so it works for macOS as well.

But, for windows still need to make .dll format for it.

Copy link
Member

@ivan-pi ivan-pi Sep 22, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This thread on SO seems to indicate .dylib as the canonical "dynamic library", whereas ".so" is used for plugins. In practice the formats are (mostly) the same. The Mac Dynamic Library Usage Guidelines refer to .dylib in many places, but don't mention .so.

So I think .dylib and the flag -dynamiclib instead of -shared would be the preferred option for MacOS.

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree with @perazz, this command line requires some branching depending on the OS and the compiler. In this discussion I had written a few versions for (gfortran/ifort) (Windows/Linux) #655 (comment) :

gfortran + linux

>gfortran -cpp -O3 -fpic -c <name>.f90
>gfortran -shared -o lib<name>.so <name>.o

gfortran + windows

>gfortran -cpp -O3 -fpic -c <name>.f90
>gfortran -shared -static -o lib<name>.so <name>.o

ifort + linux

>ifort -fpp -O3 -fpic -c <name>.f90
>ifort -shared -o lib<name>.so <name>.o

ifort + windows

>ifort -cpp -O3 -fpic -c <name>.f90
>ifort -dll -exe:<name>.dll <name>.obj

It would be needed to check how this changed with ifx. A couple of attention points, for building the shared library, the static object would have had to be built with the equivalent of -fPIC. This should be either be written as a Note/Warning or forced internally.

So, instead of hard writting ".so" in the command line. Have a character string varaible with the different names ".so" (Linux), ".dll" (Windows), "(.dylib)" macOS.

It might be interesting to split the command line in parametrizable parts:
shared_command, shared_suffix. That way, the command can also be adjusted, ifort on windows uses a different nomenclature. As shown by @ivan-pi macOS also. The command can be augmented with gfortran using -static to indicate building a fatter binary to avoid extra runtime dependencies.

// output_file, echo=self%echo, verbose=self%verbose)
end subroutine generate_shared_library


!> Compile a C object
subroutine compile_c(self, input, output, args, log_file, stat)
Expand Down
Loading