2024-04-15 15:35:35 +02:00
|
|
|
# MIT License
|
|
|
|
#
|
|
|
|
# Copyright The SCons Foundation
|
|
|
|
#
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
# a copy of this software and associated documentation files (the
|
|
|
|
# "Software"), to deal in the Software without restriction, including
|
|
|
|
# without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
# distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
# permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
# the following conditions:
|
|
|
|
#
|
|
|
|
# The above copyright notice and this permission notice shall be included
|
|
|
|
# in all copies or substantial portions of the Software.
|
|
|
|
#
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
|
|
|
|
# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
|
|
|
# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
2023-06-11 12:31:23 +02:00
|
|
|
"""SCons.Tool.cyglink
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
Customization of gnulink for Cygwin (https://www.cygwin.com/)
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
There normally shouldn't be any need to import this module directly.
|
|
|
|
It will usually be imported through the generic SCons.Tool.Tool()
|
|
|
|
selection method.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
from SCons.Tool.linkCommon import StringizeLibSymlinks, EmitLibSymlinks
|
|
|
|
from SCons.Util import CLVar, is_String
|
|
|
|
from . import gnulink
|
|
|
|
|
|
|
|
|
|
|
|
def cyglink_lib_emitter(target, source, env, **kw):
|
|
|
|
verbose = True
|
|
|
|
|
|
|
|
if 'variable_prefix' in kw:
|
|
|
|
var_prefix = kw['variable_prefix']
|
|
|
|
else:
|
|
|
|
var_prefix = 'SHLIB'
|
|
|
|
|
|
|
|
no_import_lib = env.get('no_import_lib', False)
|
|
|
|
|
|
|
|
if verbose:
|
2024-04-15 15:35:35 +02:00
|
|
|
print(f"cyglink_lib_emitter: target[0]={target[0].get_path()!r}")
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
if not no_import_lib:
|
|
|
|
# Specify import lib and add to targets
|
|
|
|
|
|
|
|
import_lib = env.subst('$%s_IMPLIBNAME' % var_prefix, target=target, source=source)
|
|
|
|
import_lib_target = env.fs.File(import_lib)
|
2024-04-15 15:35:35 +02:00
|
|
|
import_lib_target.attributes.shared = True
|
2023-06-11 12:31:23 +02:00
|
|
|
target.append(import_lib_target)
|
|
|
|
|
|
|
|
if verbose:
|
2024-04-15 15:35:35 +02:00
|
|
|
print(f"cyglink_lib_emitter: import_lib={import_lib}")
|
2023-06-11 12:31:23 +02:00
|
|
|
print("cyglink_lib_emitter: target=%s" % target)
|
|
|
|
|
|
|
|
for tgt in target:
|
|
|
|
if is_String(tgt):
|
|
|
|
tgt = env.File(tgt)
|
2024-04-15 15:35:35 +02:00
|
|
|
tgt.attributes.shared = True
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
return target, source
|
|
|
|
|
|
|
|
|
|
|
|
def cyglink_ldmodule_emitter(target, source, env, **kw):
|
|
|
|
return cyglink_lib_emitter(target, source, env, variable_prefix='LDMODULE')
|
|
|
|
|
|
|
|
|
|
|
|
def cyglink_shlib_symlink_emitter(target, source, env, **kw):
|
|
|
|
"""
|
|
|
|
On cygwin, we only create a symlink from the non-versioned implib to the versioned implib.
|
|
|
|
We don't version the shared library itself.
|
|
|
|
:param target:
|
|
|
|
:param source:
|
|
|
|
:param env:
|
|
|
|
:param kw:
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
verbose = True
|
|
|
|
|
|
|
|
if 'variable_prefix' in kw:
|
|
|
|
var_prefix = kw['variable_prefix']
|
|
|
|
else:
|
|
|
|
var_prefix = 'SHLIB'
|
|
|
|
|
|
|
|
no_import_lib = env.get('no_import_lib', False)
|
|
|
|
if no_import_lib in ['1', 'True', 'true', True]:
|
|
|
|
if verbose:
|
|
|
|
print("cyglink_shlib_symlink_emitter: no_import_lib=%s" % no_import_lib)
|
|
|
|
return target, source
|
|
|
|
|
|
|
|
no_symlinks = env.subst('$%sNOVERSIONSYMLINKS' % var_prefix)
|
|
|
|
if no_symlinks in ['1', 'True', 'true', True]:
|
|
|
|
return target, source
|
|
|
|
|
|
|
|
shlibversion = env.subst('$%sVERSION' % var_prefix)
|
|
|
|
if shlibversion:
|
|
|
|
if verbose:
|
|
|
|
print("cyglink_shlib_symlink_emitter: %sVERSION=%s" % (var_prefix, shlibversion))
|
|
|
|
|
|
|
|
# The implib (added by the cyglink_lib_emitter)
|
|
|
|
imp_lib_node = target[1]
|
|
|
|
shlib_noversion_symlink = env.subst('$%s_NOVERSION_SYMLINK' % var_prefix, target=target[0], source=source)
|
|
|
|
|
|
|
|
if verbose:
|
|
|
|
print("cyglink_shlib_symlink_emitter: shlib_noversion_symlink :%s" % shlib_noversion_symlink)
|
|
|
|
print("cyglink_shlib_symlink_emitter: imp_lib_node :%s" % imp_lib_node)
|
|
|
|
|
|
|
|
symlinks = [(env.File(shlib_noversion_symlink), imp_lib_node)]
|
|
|
|
|
|
|
|
if verbose:
|
|
|
|
print("cyglink_shlib_symlink_emitter: symlinks={!r}".format(
|
|
|
|
', '.join(["%r->%r" % (k, v) for k, v in StringizeLibSymlinks(symlinks)])
|
|
|
|
))
|
|
|
|
|
|
|
|
if symlinks:
|
|
|
|
# This does the actual symlinking
|
|
|
|
EmitLibSymlinks(env, symlinks, target[0])
|
|
|
|
|
|
|
|
# This saves the information so if the versioned shared library is installed
|
|
|
|
# it can faithfully reproduce the correct symlinks
|
|
|
|
target[0].attributes.shliblinks = symlinks
|
|
|
|
|
|
|
|
return target, source
|
|
|
|
|
|
|
|
|
|
|
|
def cyglink_ldmod_symlink_emitter(target, source, env, **kw):
|
|
|
|
return cyglink_shlib_symlink_emitter(target, source, env, variable_prefix='LDMODULE')
|
|
|
|
|
|
|
|
|
|
|
|
def cyglink_shlibversion(target, source, env, for_signature):
|
|
|
|
var_prefix = 'SHLIB'
|
|
|
|
var = '%sVERSION' % var_prefix
|
|
|
|
if var not in env:
|
|
|
|
return ''
|
|
|
|
|
|
|
|
version = env.subst("$%s" % var, target=target, source=source)
|
|
|
|
version = version.replace('.', '-')
|
|
|
|
return "." + version
|
|
|
|
|
|
|
|
|
|
|
|
def cyglink_ldmodule_version(target, source, env, for_signature):
|
|
|
|
var_prefix = 'LDMODULE'
|
|
|
|
var = '%sVERSION' % var_prefix
|
|
|
|
if var not in env:
|
|
|
|
return ''
|
|
|
|
|
|
|
|
version = env.subst("$%s" % var, target=target, source=source)
|
|
|
|
version = version.replace('.', '-')
|
|
|
|
return "." + version
|
|
|
|
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def _implib_pre_flags(target, source, env, for_signature) -> str:
|
2023-06-11 12:31:23 +02:00
|
|
|
no_import_lib = env.get('no_import_lib', False)
|
|
|
|
if no_import_lib in ['1', 'True', 'true', True]:
|
|
|
|
return ''
|
|
|
|
else:
|
|
|
|
return '-Wl,--out-implib=${TARGETS[1]} -Wl,--export-all-symbols -Wl,--enable-auto-import -Wl,--whole-archive'
|
|
|
|
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def _implib_post_flags(target, source, env, for_signature) -> str:
|
2023-06-11 12:31:23 +02:00
|
|
|
no_import_lib = env.get('no_import_lib', False)
|
|
|
|
if no_import_lib in ['1', 'True', 'true', True]:
|
|
|
|
return ''
|
|
|
|
else:
|
|
|
|
return '-Wl,--no-whole-archive'
|
|
|
|
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def generate(env) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
"""Add Builders and construction variables for cyglink to an Environment."""
|
|
|
|
gnulink.generate(env)
|
|
|
|
|
|
|
|
env['LINKFLAGS'] = CLVar('-Wl,-no-undefined')
|
|
|
|
|
|
|
|
env['SHLIBPREFIX'] = 'cyg'
|
|
|
|
env['SHLIBSUFFIX'] = '.dll'
|
|
|
|
|
|
|
|
env['IMPLIBPREFIX'] = 'lib'
|
|
|
|
env['IMPLIBSUFFIX'] = '.dll.a'
|
|
|
|
|
|
|
|
# Variables used by versioned shared libraries
|
|
|
|
# SHLIBVERSIONFLAGS and LDMODULEVERSIONFLAGS are same as in gnulink...
|
|
|
|
env['_SHLIBVERSIONFLAGS'] = '$SHLIBVERSIONFLAGS'
|
|
|
|
env['_LDMODULEVERSIONFLAGS'] = '$LDMODULEVERSIONFLAGS'
|
|
|
|
|
|
|
|
env['_IMPLIB_PRE_SOURCES'] = _implib_pre_flags
|
|
|
|
env['_IMPLIB_POST_SOURCES'] = _implib_post_flags
|
|
|
|
env['SHLINKCOM'] = '$SHLINK -o $TARGET $SHLINKFLAGS $__SHLIBVERSIONFLAGS $__RPATH ' \
|
|
|
|
'$_IMPLIB_PRE_SOURCES $SOURCES $_IMPLIB_POST_SOURCES $_LIBDIRFLAGS $_LIBFLAGS'
|
|
|
|
env['LDMODULECOM'] = '$LDMODULE -o $TARGET $SHLINKFLAGS $__LDMODULEVERSIONFLAGS $__RPATH ' \
|
|
|
|
'$_IMPLIB_PRE_SOURCES $SOURCES $_IMPLIB_POST_SOURCES $_LIBDIRFLAGS $_LIBFLAGS'
|
|
|
|
|
|
|
|
# Overwrite emitters. Cyglink does things differently when creating symlinks
|
|
|
|
env['SHLIBEMITTER'] = [cyglink_lib_emitter, cyglink_shlib_symlink_emitter]
|
|
|
|
env['LDMODULEEMITTER'] = [cyglink_ldmodule_emitter, cyglink_ldmod_symlink_emitter]
|
|
|
|
|
|
|
|
# This is the non versioned shlib filename
|
|
|
|
# If SHLIBVERSION is defined then this will symlink to $SHLIBNAME
|
|
|
|
env['SHLIB_NOVERSION_SYMLINK'] = '${IMPLIBPREFIX}$_get_shlib_stem${IMPLIBSUFFIX}'
|
|
|
|
env['LDMODULE_NOVERSION_SYMLINK'] = '${IMPLIBPREFIX}$_get_ldmodule_stem${IMPLIBSUFFIX}'
|
|
|
|
|
|
|
|
env['SHLIB_IMPLIBNAME'] = '${IMPLIBPREFIX}$_get_shlib_stem${_SHLIB_IMPLIBSUFFIX}'
|
|
|
|
env['LDMODULE_IMPLIBNAME'] = '${IMPLIBPREFIX}$_get_ldmodule_stem${_LDMODULE_IMPLIBSUFFIX}'
|
|
|
|
|
|
|
|
env['_cyglink_shlibversion'] = cyglink_shlibversion
|
|
|
|
env['_SHLIB_IMPLIBSUFFIX'] = '${_cyglink_shlibversion}${IMPLIBSUFFIX}'
|
|
|
|
env['_SHLIBSUFFIX'] = '${_cyglink_shlibversion}${SHLIBSUFFIX}'
|
|
|
|
|
|
|
|
env['_cyglink_ldmodule_version'] = cyglink_ldmodule_version
|
|
|
|
|
|
|
|
env['_LDMODULESUFFIX'] = '${_cyglink_ldmodule_version}${LDMODULESUFFIX}'
|
|
|
|
env['_LDMODULE_IMPLIBSUFFIX'] = '${_cyglink_ldmodule_version}${IMPLIBSUFFIX}'
|
|
|
|
|
|
|
|
# Remove variables set by default initialization which aren't needed/used by cyglink
|
|
|
|
# these variables were set by gnulink but are not used in cyglink
|
|
|
|
for rv in ['_SHLIBSONAME', '_LDMODULESONAME']:
|
|
|
|
if rv in env:
|
|
|
|
del env[rv]
|
|
|
|
|
|
|
|
|
|
|
|
def exists(env):
|
|
|
|
return gnulink.exists(env)
|
|
|
|
|
|
|
|
# Local Variables:
|
|
|
|
# tab-width:4
|
|
|
|
# indent-tabs-mode:nil
|
|
|
|
# End:
|
|
|
|
# vim: set expandtab tabstop=4 shiftwidth=4:
|