Blame src/logic/slbt_exec_link.c

2bd749
/*******************************************************************/
2bd749
/*  slibtool: a skinny libtool implementation, written in C        */
2bd749
/*  Copyright (C) 2016  Z. Gilboa                                  */
2bd749
/*  Released under the Standard MIT License; see COPYING.SLIBTOOL. */
2bd749
/*******************************************************************/
2bd749
8b7d50
#include <stdlib.h>
f0d5f3
#include <stdio.h>
2bd749
#include <string.h>
2bd749
#include <stdbool.h>
2bd749
#include <fcntl.h>
2bd749
#include <errno.h>
2bd749
#include <sys/stat.h>
2bd749
2bd749
#include <slibtool/slibtool.h>
2bd749
#include "slibtool_spawn_impl.h"
6529aa
#include "slibtool_symlink_impl.h"
2bd749
8b7d50
struct slbt_deps_meta {
8b7d50
	char ** altv;
8b7d50
	char *	args;
8b7d50
	int	depscnt;
8b7d50
	int	infolen;
8b7d50
};
8b7d50
5cc3b3
/*******************************************************************/
5cc3b3
/*                                                                 */
5cc3b3
/* -o <ltlib>  switches              input   result                */
5cc3b3
/* ----------  --------------------- -----   ------                */
b07789
/* libfoo.a    [-shared|-static]     bar.lo  libfoo.a              */
5cc3b3
/*                                                                 */
6d048d
/* ar cru libfoo.a bar.o                                           */
5cc3b3
/* ranlib libfoo.a                                                 */
5cc3b3
/*                                                                 */
5cc3b3
/*******************************************************************/
5cc3b3
a0c318
/*******************************************************************/
a0c318
/*                                                                 */
a0c318
/* -o <ltlib>  switches              input   result                */
a0c318
/* ----------  --------------------- -----   ------                */
a0c318
/* libfoo.la   -shared               bar.lo  libfoo.la             */
a0c318
/*                                           .libs/libfoo.a        */
a0c318
/*                                           .libs/libfoo.la (lnk) */
a0c318
/*                                                                 */
a0c318
/* ar cru .libs/libfoo.a .libs/bar.o                               */
a0c318
/* ranlib .libs/libfoo.a                                           */
a0c318
/* (generate libfoo.la)                                            */
a0c318
/* ln -s ../libfoo.la .libs/libfoo.la                              */
f730ac
/*                                                                 */
f730ac
/*******************************************************************/
f730ac
f730ac
/*******************************************************************/
f730ac
/*                                                                 */
f730ac
/* -o <ltlib>  switches              input   result                */
f730ac
/* ----------  --------------------- -----   ------                */
f730ac
/* libfoo.la   -static               bar.lo  libfoo.la             */
f730ac
/*                                           .libs/libfoo.a        */
f730ac
/*                                           .libs/libfoo.la (lnk) */
f730ac
/*                                                                 */
f730ac
/* ar cru .libs/libfoo.a bar.o                                     */
f730ac
/* ranlib .libs/libfoo.a                                           */
f730ac
/* (generate libfoo.la)                                            */
f730ac
/* ln -s ../libfoo.la .libs/libfoo.la                              */
a0c318
/*                                                                 */
a0c318
/*******************************************************************/
a0c318
3be47d
static int slbt_exec_link_exit(
3be47d
	struct slbt_deps_meta *	depsmeta,
3be47d
	int			ret)
3be47d
{
3be47d
	if (depsmeta->altv)
3be47d
		free(depsmeta->altv);
3be47d
3be47d
	if (depsmeta->args)
3be47d
		free(depsmeta->args);
3be47d
3be47d
	return ret;
3be47d
}
3be47d
8b7d50
static int slbt_get_deps_meta(
8b7d50
	char *			libfilename,
8b7d50
	struct slbt_deps_meta *	depsmeta)
8b7d50
{
8b7d50
	int		ret;
8b7d50
	FILE *		fdeps;
8b7d50
	struct stat	st;
8b7d50
	char		depfile[4*PATH_MAX];
8b7d50
	char *		deplibs = depfile;
8b7d50
8b7d50
	if ((size_t)snprintf(depfile,sizeof(depfile),"%s.slibtool.deps",
8b7d50
				libfilename)
8b7d50
			>= sizeof(depfile))
8b7d50
		return -1;
8b7d50
8b7d50
	if ((stat(depfile,&st)))
8b7d50
		return -1;
8b7d50
8b7d50
	if (!(fdeps = fopen(depfile,"r")))
8b7d50
		return -1;
8b7d50
8b7d50
	if ((size_t)st.st_size >= sizeof(depfile))
8b7d50
		if (!(deplibs = malloc(st.st_size+1))) {
8b7d50
			fclose(fdeps);
8b7d50
			return -1;
8b7d50
		}
8b7d50
8b7d50
	depsmeta->infolen += st.st_size;
8b7d50
	depsmeta->infolen++;
8b7d50
8b7d50
	while (fscanf(fdeps,"%s\n",deplibs) == 1)
8b7d50
		depsmeta->depscnt++;
8b7d50
8b7d50
	ret = ferror(fdeps) ? -1 : 0;
8b7d50
	fclose(fdeps);
8b7d50
8b7d50
	return ret;
8b7d50
}
8b7d50
14d60a
static bool slbt_adjust_input_argument(
14d60a
	char *		arg,
14d60a
	const char *	osuffix,
14d60a
	const char *	asuffix,
14d60a
	bool		fpic)
5cc3b3
{
53aa85
	char *	slash;
5cc3b3
	char *	dot;
53aa85
	char	base[PATH_MAX];
5cc3b3
5cc3b3
	if (*arg == '-')
5cc3b3
		return false;
5cc3b3
5cc3b3
	if (!(dot = strrchr(arg,'.')))
5cc3b3
		return false;
5cc3b3
14d60a
	if (strcmp(dot,osuffix))
5cc3b3
		return false;
5cc3b3
5cc3b3
	if (fpic) {
53aa85
		if ((slash = strrchr(arg,'/')))
53aa85
			slash++;
53aa85
		else
53aa85
			slash = arg;
53aa85
53aa85
		if ((size_t)snprintf(base,sizeof(base),"%s",
53aa85
				slash) >= sizeof(base))
53aa85
			return false;
53aa85
53aa85
		sprintf(slash,".libs/%s",base);
53aa85
		dot = strrchr(arg,'.');
5cc3b3
	}
53aa85
14d60a
	strcpy(dot,asuffix);
53aa85
	return true;
5cc3b3
}
5cc3b3
afaba2
static int slbt_adjust_linker_argument(
b3940a
	char *		arg,
b3940a
	bool		fpic,
b3940a
	const char *	dsosuffix,
8b7d50
	const char *	arsuffix,
8b7d50
	struct slbt_deps_meta * depsmeta)
b3940a
{
b3940a
	int	fdlib;
b3940a
	char *	slash;
b3940a
	char *	dot;
b3940a
	char	base[PATH_MAX];
b3940a
b3940a
	if (*arg == '-')
afaba2
		return 0;
b3940a
b3940a
	if (!(dot = strrchr(arg,'.')))
afaba2
		return 0;
b3940a
b3940a
	if (strcmp(dot,".la"))
afaba2
		return 0;
b3940a
b3940a
	if (fpic) {
b3940a
		if ((slash = strrchr(arg,'/')))
b3940a
			slash++;
b3940a
		else
b3940a
			slash = arg;
b3940a
b3940a
		if ((size_t)snprintf(base,sizeof(base),"%s",
b3940a
				slash) >= sizeof(base))
afaba2
			return 0;
b3940a
b3940a
		sprintf(slash,".libs/%s",base);
b3940a
		dot = strrchr(arg,'.');
b3940a
	}
b3940a
b3940a
	/* shared library dependency? */
b3940a
	if (fpic) {
b3940a
		sprintf(dot,"%s",dsosuffix);
b3940a
b3940a
		if ((fdlib = open(arg,O_RDONLY)) >= 0)
b3940a
			close(fdlib);
b3940a
		else
b3940a
			sprintf(dot,"%s",arsuffix);
b3940a
8b7d50
		return slbt_get_deps_meta(arg,depsmeta);
b3940a
	}
b3940a
b3940a
	/* input archive */
b3940a
	sprintf(dot,"%s",arsuffix);
afaba2
	return 0;
b3940a
}
b3940a
2ef5be
static int slbt_exec_link_adjust_argument_vector(
2ef5be
	const struct slbt_driver_ctx *	dctx,
2ef5be
	struct slbt_exec_ctx *		ectx,
3be47d
	struct slbt_deps_meta *		depsmeta,
79c501
	const char *			cwd,
2ef5be
	bool				flibrary)
2ef5be
{
2ef5be
	char ** carg;
2ef5be
	char ** aarg;
2ef5be
	char *	slash;
2ef5be
	char *	mark;
3be47d
	char *	darg;
2ef5be
	char *	dot;
3be47d
	FILE *	fdeps;
3be47d
	char *	dpath;
3be47d
	int	argc;
2ef5be
	char	arg[PATH_MAX];
3be47d
	char	lib[PATH_MAX];
2ef5be
	bool	fwholearchive = false;
2ef5be
3be47d
	for (argc=0,carg=ectx->cargv; *carg; carg++)
3be47d
		argc++;
3be47d
3be47d
	if (!(depsmeta->args = calloc(1,depsmeta->infolen)))
3be47d
		return -1;
3be47d
3be47d
	argc *= 3;
3be47d
	argc += depsmeta->depscnt;
3be47d
3be47d
	if (!(depsmeta->altv = calloc(argc,sizeof(char *))))
3be47d
		return -1;
3be47d
2ef5be
	carg = ectx->cargv;
3be47d
	aarg = depsmeta->altv;
3be47d
	darg = depsmeta->args;
2ef5be
2ef5be
	for (; *carg; ) {
3be47d
		dpath = 0;
3be47d
2ef5be
		if (!strcmp(*carg,"-Wl,--whole-archive"))
2ef5be
			fwholearchive = true;
2ef5be
		else if (!strcmp(*carg,"-Wl,--no-whole-archive"))
2ef5be
			fwholearchive = false;
2ef5be
2ef5be
2ef5be
2ef5be
		if (**carg == '-') {
2ef5be
			*aarg++ = *carg++;
2ef5be
2ef5be
		} else if (!(dot = strrchr(*carg,'.'))) {
2ef5be
			*aarg++ = *carg++;
2ef5be
2ef5be
		} else if (!(strcmp(dot,".a"))) {
2ef5be
			if (flibrary && !fwholearchive)
2ef5be
				*aarg++ = "-Wl,--whole-archive";
2ef5be
3be47d
			dpath = lib;
3be47d
			sprintf(lib,"%s.slibtool.deps",*carg);
2ef5be
			*aarg++ = *carg++;
2ef5be
2ef5be
			if (flibrary && !fwholearchive)
2ef5be
				*aarg++ = "-Wl,--no-whole-archive";
2ef5be
2ef5be
		} else if (strcmp(dot,dctx->cctx->settings.dsosuffix)) {
2ef5be
			*aarg++ = *carg++;
2ef5be
867d1e
		} else if (carg == ectx->lout[1]) {
867d1e
			/* ^^^hoppla^^^ */
867d1e
			*aarg++ = *carg++;
2ef5be
		} else {
3be47d
			dpath = lib;
3be47d
			sprintf(lib,"%s.slibtool.deps",*carg);
3be47d
2ef5be
			/* account for {'-','L','-','l'} */
2ef5be
			if ((size_t)snprintf(arg,sizeof(arg),"%s",
2ef5be
					*carg) >= (sizeof(arg) - 4))
2ef5be
				return -1;
2ef5be
2ef5be
			if ((slash = strrchr(arg,'/'))) {
2ef5be
				sprintf(*carg,"-L%s",arg);
2ef5be
2ef5be
				mark   = strrchr(*carg,'/');
2ef5be
				*mark  = '\0';
2ef5be
5ee0d1
				if (ectx->fwrapper) {
5ee0d1
					*slash = '\0';
5ee0d1
5ee0d1
					if (fprintf(ectx->fwrapper,
79c501
							"DL_PATH=\"$DL_PATH$COLON%s/%s\"\n"
5ee0d1
							"COLON=':'\n\n",
79c501
							cwd,arg) < 0)
5ee0d1
						return -1;
5ee0d1
				}
5ee0d1
2ef5be
				*aarg++ = *carg++;
2ef5be
				*aarg++ = ++mark;
2ef5be
2ef5be
				++slash;
2ef5be
				slash += strlen(dctx->cctx->settings.dsoprefix);
2ef5be
2ef5be
				sprintf(mark,"-l%s",slash);
2ef5be
				dot  = strrchr(mark,'.');
2ef5be
				*dot = '\0';
2ef5be
			} else {
2ef5be
				*aarg++ = *carg++;
2ef5be
			}
2ef5be
		}
3be47d
3be47d
		if (dpath) {
3be47d
			*aarg = darg;
3be47d
3be47d
			if (!(fdeps = fopen(dpath,"r"))) {
3be47d
				free(depsmeta->altv);
3be47d
				free(depsmeta->args);
3be47d
				return -1;
3be47d
			}
3be47d
3be47d
			while (fscanf(fdeps,"%s\n",darg) == 1) {
3be47d
				*aarg++ = darg;
3be47d
				darg   += strlen(darg) + sizeof('\0');
3be47d
			}
3be47d
3be47d
			if (ferror(fdeps)) {
3be47d
				free(depsmeta->altv);
3be47d
				free(depsmeta->args);
3be47d
				fclose(fdeps);
3be47d
				return -1;
3be47d
			} else {
3be47d
				fclose(fdeps);
3be47d
			}
3be47d
		}
2ef5be
	}
2ef5be
2ef5be
	return 0;
2ef5be
}
2ef5be
628cde
static int slbt_exec_link_remove_file(
628cde
	const struct slbt_driver_ctx *	dctx,
628cde
	struct slbt_exec_ctx *		ectx,
628cde
	const char *			target)
628cde
{
628cde
	/* remove target (if any) */
628cde
	if (!(unlink(target)) || (errno == ENOENT))
628cde
		return 0;
628cde
628cde
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT))
628cde
		strerror(errno);
628cde
628cde
	return -1;
628cde
}
628cde
0913c3
static int slbt_exec_link_create_dep_file(
0913c3
	struct slbt_exec_ctx *	ectx,
0913c3
	char **			altv,
0913c3
	const char *		libfilename)
0913c3
{
0913c3
	char **	parg;
0913c3
	char *	popt;
0913c3
	char *	plib;
0913c3
	char	depfile[PATH_MAX];
0913c3
0913c3
	if (ectx->fdeps)
0913c3
		fclose(ectx->fdeps);
0913c3
0913c3
	if ((size_t)snprintf(depfile,sizeof(depfile),"%s.slibtool.deps",
0913c3
				libfilename)
0913c3
			>= sizeof(depfile))
0913c3
		return -1;
0913c3
0913c3
	if (!(ectx->fdeps = fopen(depfile,"w")))
0913c3
		return -1;
0913c3
0913c3
	for (parg=altv; *parg; parg++) {
0913c3
		popt = 0;
0913c3
		plib = 0;
0913c3
0913c3
		if (!strcmp(*parg,"-l")) {
0913c3
			popt = *parg++;
0913c3
			plib = *parg;
0913c3
		} else if (!strcmp(*parg,"--library")) {
0913c3
			popt = *parg++;
0913c3
			plib = *parg;
0913c3
		} else if (!strncmp(*parg,"-l",2)) {
0913c3
			popt = *parg;
0913c3
			plib = popt + 2;
0913c3
		} else if (!strncmp(*parg,"--library=",10)) {
0913c3
			popt = *parg;
0913c3
			plib = popt + 10;
0913c3
		}
0913c3
0913c3
		if (plib)
0913c3
			if (fprintf(ectx->fdeps,"-l%s\n",plib) < 0)
0913c3
				return -1;
0913c3
	}
0913c3
0913c3
	if (fflush(ectx->fdeps))
0913c3
		return -1;
0913c3
0913c3
	return 0;
0913c3
}
0913c3
3b5e3d
static int slbt_exec_link_create_archive(
5cc3b3
	const struct slbt_driver_ctx *	dctx,
a51ace
	struct slbt_exec_ctx *		ectx,
3b5e3d
	const char *			arfilename,
3b5e3d
	bool				fpic)
5cc3b3
{
5cc3b3
	char ** 	aarg;
5cc3b3
	char ** 	parg;
5cc3b3
	char *		ranlib[3];
3c594d
	char		program[PATH_MAX];
3c594d
	char		output [PATH_MAX];
5cc3b3
e7ddb2
	/* initial state */
e7ddb2
	slbt_reset_arguments(ectx);
e7ddb2
5cc3b3
	/* placeholders */
5cc3b3
	slbt_reset_placeholders(ectx);
5cc3b3
5cc3b3
	/* alternate program (ar, ranlib) */
5cc3b3
	ectx->program = program;
5cc3b3
a51ace
	/* output */
a51ace
	if ((size_t)snprintf(output,sizeof(output),"%s",
a51ace
			arfilename) >= sizeof(output))
a51ace
		return -1;
a51ace
5cc3b3
	/* ar alternate argument vector */
5cc3b3
	if ((size_t)snprintf(program,sizeof(program),"%s",
5cc3b3
			dctx->cctx->host.ar) >= sizeof(program))
5cc3b3
		return -1;
5cc3b3
5cc3b3
	aarg    = ectx->altv;
5cc3b3
	*aarg++ = program;
5cc3b3
	*aarg++ = "cru";
a51ace
	*aarg++ = output;
5cc3b3
5cc3b3
	/* input argument adjustment */
5cc3b3
	for (parg=ectx->cargv; *parg; parg++)
14d60a
		if (slbt_adjust_input_argument(*parg,".lo",".o",fpic))
5cc3b3
			*aarg++ = *parg;
5cc3b3
5cc3b3
	*aarg = 0;
5cc3b3
	ectx->argv = ectx->altv;
5cc3b3
5cc3b3
	/* step output */
5cc3b3
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT))
5cc3b3
		if (slbt_output_link(dctx,ectx))
5cc3b3
			return -1;
5cc3b3
628cde
	/* remove old archive as needed */
628cde
	if (slbt_exec_link_remove_file(dctx,ectx,output))
628cde
		return -1;
628cde
0913c3
	/* .deps */
0913c3
	if (slbt_exec_link_create_dep_file(ectx,ectx->cargv,arfilename))
0913c3
		return -1;
0913c3
5cc3b3
	/* ar spawn */
5cc3b3
	if ((slbt_spawn(ectx,true) < 0) || ectx->exitcode)
5cc3b3
		return -1;
69bc4a
69bc4a
	/* input objects associated with .la archives */
69bc4a
	for (parg=ectx->cargv; *parg; parg++)
69bc4a
		if (slbt_adjust_input_argument(*parg,".la",".a",fpic))
69bc4a
			if (slbt_archive_import(dctx,ectx,output,*parg))
69bc4a
				return -1;
5cc3b3
5cc3b3
	/* ranlib argv */
5cc3b3
	if ((size_t)snprintf(program,sizeof(program),"%s",
5cc3b3
			dctx->cctx->host.ranlib) >= sizeof(program))
5cc3b3
		return -1;
5cc3b3
5cc3b3
	ranlib[0] = program;
a51ace
	ranlib[1] = output;
5cc3b3
	ranlib[2] = 0;
5cc3b3
	ectx->argv = ranlib;
5cc3b3
5cc3b3
	/* step output */
5cc3b3
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT))
5cc3b3
		if (slbt_output_link(dctx,ectx))
5cc3b3
			return -1;
5cc3b3
5cc3b3
	/* ranlib spawn */
5cc3b3
	if ((slbt_spawn(ectx,true) < 0) || ectx->exitcode)
5cc3b3
		return -1;
5cc3b3
5cc3b3
	return 0;
5cc3b3
}
5cc3b3
b3940a
static int slbt_exec_link_create_library(
b3940a
	const struct slbt_driver_ctx *	dctx,
b3940a
	struct slbt_exec_ctx *		ectx,
b3940a
	const char *			dsofilename)
b3940a
{
b3940a
	char ** parg;
79c501
	char	cwd    [PATH_MAX];
b3940a
	char	output [PATH_MAX];
0f8591
	char	soname [PATH_MAX];
8b7d50
	struct slbt_deps_meta depsmeta = {0};
b3940a
b3940a
	/* initial state */
b3940a
	slbt_reset_arguments(ectx);
b3940a
b3940a
	/* placeholders */
b3940a
	slbt_reset_placeholders(ectx);
b3940a
b3940a
	/* input argument adjustment */
b3940a
	for (parg=ectx->cargv; *parg; parg++)
14d60a
		slbt_adjust_input_argument(*parg,".lo",".o",true);
b3940a
b3940a
	/* linker argument adjustment */
b3940a
	for (parg=ectx->cargv; *parg; parg++)
afaba2
		if (slbt_adjust_linker_argument(
afaba2
				*parg,true,
afaba2
				dctx->cctx->settings.dsosuffix,
8b7d50
				dctx->cctx->settings.arsuffix,
8b7d50
				&depsmeta) < 0)
afaba2
			return -1;
b3940a
b3940a
	/* --no-undefined */
b3940a
	if (dctx->cctx->drvflags & SLBT_DRIVER_NO_UNDEFINED)
b3940a
		*ectx->noundef = "-Wl,--no-undefined";
b3940a
0f8591
	/* -soname */
867d1e
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_AVOID_VERSION)) {
867d1e
		if ((size_t)snprintf(soname,sizeof(soname),"-Wl,%s%s%s.%d",
867d1e
					dctx->cctx->settings.dsoprefix,
867d1e
					dctx->cctx->libname,
867d1e
					dctx->cctx->settings.dsosuffix,
867d1e
					dctx->cctx->verinfo.major)
867d1e
				>= sizeof(soname))
867d1e
			return -1;
0f8591
867d1e
		*ectx->soname  = "-Wl,-soname";
867d1e
		*ectx->lsoname = soname;
867d1e
	}
0f8591
b3940a
	/* shared object */
b3940a
	*ectx->dpic = "-shared";
b3940a
	*ectx->fpic = "-fPIC";
b3940a
b3940a
	/* output */
867d1e
	if (dctx->cctx->drvflags & SLBT_DRIVER_AVOID_VERSION) {
867d1e
		strcpy(output,dsofilename);
867d1e
	} else {
867d1e
		if ((size_t)snprintf(output,sizeof(output),"%s.%d.%d.%d",
867d1e
					dsofilename,
867d1e
					dctx->cctx->verinfo.major,
867d1e
					dctx->cctx->verinfo.minor,
867d1e
					dctx->cctx->verinfo.revision)
867d1e
				>= sizeof(output))
867d1e
			return -1;
867d1e
	}
b3940a
b3940a
	*ectx->lout[0] = "-o";
b3940a
	*ectx->lout[1] = output;
b3940a
79c501
	/* cwd */
79c501
	if (!getcwd(cwd,sizeof(cwd)))
79c501
		return -1;
79c501
2ef5be
	/* .libs/libfoo.so --> -L.libs -lfoo */
2ef5be
	if (slbt_exec_link_adjust_argument_vector(
3be47d
			dctx,ectx,&depsmeta,cwd,true))
2ef5be
		return -1;
2ef5be
2ef5be
	/* using alternate argument vector */
3be47d
	ectx->argv    = depsmeta.altv;
3be47d
	ectx->program = depsmeta.altv[0];
2ef5be
b3940a
	/* step output */
b3940a
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT))
b3940a
		if (slbt_output_link(dctx,ectx))
3be47d
			return slbt_exec_link_exit(&depsmeta,-1);
b3940a
0913c3
	/* .deps */
0913c3
	if (slbt_exec_link_create_dep_file(ectx,ectx->argv,dsofilename))
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
0913c3
b3940a
	/* spawn */
b3940a
	if ((slbt_spawn(ectx,true) < 0) || ectx->exitcode)
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
b3940a
3be47d
	return slbt_exec_link_exit(&depsmeta,0);
b3940a
}
b3940a
f6ccbe
static int slbt_exec_link_create_executable(
f6ccbe
	const struct slbt_driver_ctx *	dctx,
f6ccbe
	struct slbt_exec_ctx *		ectx,
f6ccbe
	const char *			exefilename)
f6ccbe
{
f6ccbe
	char ** parg;
5ee0d1
	char	cwd    [PATH_MAX];
f6ccbe
	char	output [PATH_MAX];
5ee0d1
	char	wrapper[PATH_MAX];
5e5804
	char	wraplnk[PATH_MAX];
238670
	bool	fabspath;
8b7d50
	struct slbt_deps_meta depsmeta = {0};
f6ccbe
f6ccbe
	/* initial state */
f6ccbe
	slbt_reset_arguments(ectx);
f6ccbe
f6ccbe
	/* placeholders */
f6ccbe
	slbt_reset_placeholders(ectx);
f6ccbe
f6ccbe
	/* input argument adjustment */
f6ccbe
	for (parg=ectx->cargv; *parg; parg++)
14d60a
		slbt_adjust_input_argument(*parg,".lo",".o",true);
f6ccbe
f6ccbe
	/* linker argument adjustment */
f6ccbe
	for (parg=ectx->cargv; *parg; parg++)
afaba2
		if (slbt_adjust_linker_argument(
afaba2
				*parg,true,
afaba2
				dctx->cctx->settings.dsosuffix,
8b7d50
				dctx->cctx->settings.arsuffix,
8b7d50
				&depsmeta) < 0)
afaba2
			return -1;
f6ccbe
f6ccbe
	/* --no-undefined */
f6ccbe
	if (dctx->cctx->drvflags & SLBT_DRIVER_NO_UNDEFINED)
f6ccbe
		*ectx->noundef = "-Wl,--no-undefined";
f6ccbe
5ee0d1
	/* executable wrapper: header */
5ee0d1
	if ((size_t)snprintf(wrapper,sizeof(wrapper),"%s.wrapper.tmp",
5ee0d1
				dctx->cctx->output)
5ee0d1
			>= sizeof(wrapper))
5ee0d1
		return -1;
5ee0d1
5ee0d1
	if (!(ectx->fwrapper = fopen(wrapper,"w")))
5ee0d1
		return -1;
5ee0d1
5ee0d1
	if (fprintf(ectx->fwrapper,
5ee0d1
			"#!/bin/sh\n"
5ee0d1
			"# slibtool (pre-alpha): generated executable wrapper\n\n"
5ee0d1
			"if [ -z \"$%s\" ]; then\n"
5ee0d1
			"\tDL_PATH=\n"
5ee0d1
			"\tCOLON=\n"
5ee0d1
			"\tLCOLON=\n"
5ee0d1
			"else\n"
5ee0d1
			"\tDL_PATH=\n"
5ee0d1
			"\tCOLON=\n"
5ee0d1
			"\tLCOLON=':'\n"
5ee0d1
			"fi\n\n",
5ee0d1
			dctx->cctx->settings.ldpathenv) < 0)
5ee0d1
		return -1;
5ee0d1
f6ccbe
	/* output */
f6ccbe
	if ((size_t)snprintf(output,sizeof(output),"%s",
f6ccbe
				exefilename)
f6ccbe
			>= sizeof(output))
f6ccbe
		return -1;
f6ccbe
f6ccbe
	*ectx->lout[0] = "-o";
f6ccbe
	*ectx->lout[1] = output;
f6ccbe
79c501
	/* cwd */
79c501
	if (!getcwd(cwd,sizeof(cwd)))
79c501
		return -1;
79c501
2ef5be
	/* .libs/libfoo.so --> -L.libs -lfoo */
2ef5be
	if (slbt_exec_link_adjust_argument_vector(
3be47d
			dctx,ectx,&depsmeta,cwd,false))
2ef5be
		return -1;
2ef5be
2ef5be
	/* using alternate argument vector */
3be47d
	ectx->argv    = depsmeta.altv;
3be47d
	ectx->program = depsmeta.altv[0];
2ef5be
5ee0d1
	/* executable wrapper: footer */
238670
	fabspath = (exefilename[0] == '/');
238670
5ee0d1
	if (fprintf(ectx->fwrapper,
5ee0d1
			"DL_PATH=\"$DL_PATH$LCOLON$%s\"\n\n"
5ee0d1
			"export %s=$DL_PATH\n\n"
5ee0d1
			"exec %s/%s \"$@\"\n",
5ee0d1
			dctx->cctx->settings.ldpathenv,
5ee0d1
			dctx->cctx->settings.ldpathenv,
238670
			fabspath ? "" : cwd,
238670
			fabspath ? &exefilename[1] : exefilename) < 0)
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
5ee0d1
f6ccbe
	/* step output */
f6ccbe
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT))
f6ccbe
		if (slbt_output_link(dctx,ectx))
3be47d
			return slbt_exec_link_exit(&depsmeta,-1);
f6ccbe
f6ccbe
	/* spawn */
f6ccbe
	if ((slbt_spawn(ectx,true) < 0) || ectx->exitcode)
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
f6ccbe
5ee0d1
	/* executable wrapper: finalize */
5ee0d1
	fclose(ectx->fwrapper);
5ee0d1
	ectx->fwrapper = 0;
5ee0d1
5e5804
	if ((size_t)snprintf(wraplnk,sizeof(wraplnk),"%s.exe.wrapper",
5e5804
			dctx->cctx->output) >= sizeof(wraplnk))
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
5e5804
5e5804
	if (slbt_create_symlink(
5e5804
			dctx,ectx,
5e5804
			dctx->cctx->output,wraplnk,
5e5804
			false))
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
5e5804
5ee0d1
	if (rename(wrapper,dctx->cctx->output))
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
5ee0d1
5ee0d1
	if (chmod(dctx->cctx->output,0755))
3be47d
		return slbt_exec_link_exit(&depsmeta,-1);
5ee0d1
3be47d
	return slbt_exec_link_exit(&depsmeta,0);
f6ccbe
}
f6ccbe
b3940a
static int slbt_exec_link_create_library_symlink(
b3940a
	const struct slbt_driver_ctx *	dctx,
b3940a
	struct slbt_exec_ctx *		ectx,
b3940a
	bool				fmajor)
b3940a
{
b3940a
	char	target[PATH_MAX];
b3940a
	char	lnkname[PATH_MAX];
b3940a
b3940a
	sprintf(target,"%s.%d.%d.%d",
b3940a
		ectx->dsofilename,
b3940a
		dctx->cctx->verinfo.major,
b3940a
		dctx->cctx->verinfo.minor,
b3940a
		dctx->cctx->verinfo.revision);
b3940a
b3940a
	if (fmajor)
b3940a
		sprintf(lnkname,"%s.%d",
b3940a
			ectx->dsofilename,
b3940a
			dctx->cctx->verinfo.major);
b3940a
b3940a
	else
b3940a
		strcpy(lnkname,ectx->dsofilename);
b3940a
6529aa
	return slbt_create_symlink(
b3940a
		dctx,ectx,
b3940a
		target,lnkname,
b3940a
		false);
b3940a
}
b3940a
2bd749
int slbt_exec_link(
2bd749
	const struct slbt_driver_ctx *	dctx,
2bd749
	struct slbt_exec_ctx *		ectx)
2bd749
{
2bd749
	int			ret;
2bd749
	int			fdlibs;
a51ace
	const char *		output;
2bd749
	char *			dot;
2bd749
	FILE *			fout;
2bd749
	struct slbt_exec_ctx *	actx;
2bd749
2bd749
	/* context */
2bd749
	if (ectx)
2bd749
		actx = 0;
2bd749
	else if ((ret = slbt_get_exec_ctx(dctx,&ectx)))
2bd749
		return ret;
2bd749
	else
2bd749
		actx = ectx;
2bd749
5cc3b3
	/* output suffix */
a51ace
	output = dctx->cctx->output;
a51ace
	dot    = strrchr(output,'.');
5cc3b3
2bd749
	/* .libs directory */
2bd749
	if (dctx->cctx->drvflags & SLBT_DRIVER_SHARED) {
2bd749
		if ((fdlibs = open(ectx->ldirname,O_DIRECTORY)) >= 0)
2bd749
			close(fdlibs);
2bd749
		else if ((errno != ENOENT) || mkdir(ectx->ldirname,0777)) {
2bd749
			slbt_free_exec_ctx(actx);
2bd749
			return -1;
2bd749
		}
2bd749
	}
2bd749
5cc3b3
	/* non-pic libfoo.a */
5cc3b3
	if (dot && !strcmp(dot,".a"))
3b5e3d
		if (slbt_exec_link_create_archive(dctx,ectx,output,false)) {
5cc3b3
			slbt_free_exec_ctx(actx);
5cc3b3
			return -1;
5cc3b3
		}
5cc3b3
a0c318
	/* pic libfoo.a */
a0c318
	if (dot && !strcmp(dot,".la"))
a0c318
		if (slbt_exec_link_create_archive(
a0c318
				dctx,ectx,
a0c318
				ectx->arfilename,
5fff27
				dctx->cctx->drvflags & SLBT_DRIVER_SHARED)) {
a0c318
			slbt_free_exec_ctx(actx);
a0c318
			return -1;
a0c318
		}
a0c318
b3940a
	/* dynamic library */
b3940a
	if (dot && !strcmp(dot,".la") && dctx->cctx->rpath) {
b3940a
		/* linking: libfoo.so.x.y.z */
b3940a
		if (slbt_exec_link_create_library(
b3940a
				dctx,ectx,
b3940a
				ectx->dsofilename)) {
b3940a
			slbt_free_exec_ctx(actx);
b3940a
			return -1;
b3940a
		}
b3940a
867d1e
		if (!(dctx->cctx->drvflags & SLBT_DRIVER_AVOID_VERSION)) {
867d1e
			/* symlink: libfoo.so.x --> libfoo.so.x.y.z */
867d1e
			if (slbt_exec_link_create_library_symlink(
867d1e
					dctx,ectx,
867d1e
					true)) {
867d1e
				slbt_free_exec_ctx(actx);
867d1e
				return -1;
867d1e
			}
b3940a
867d1e
			/* symlink: libfoo.so --> libfoo.so.x.y.z */
867d1e
			if (slbt_exec_link_create_library_symlink(
867d1e
					dctx,ectx,
867d1e
					false)) {
867d1e
				slbt_free_exec_ctx(actx);
867d1e
				return -1;
867d1e
			}
b3940a
		}
b3940a
	}
b3940a
f6ccbe
	/* executable */
f6ccbe
	if (!dctx->cctx->rpath && !dctx->cctx->libname) {
f6ccbe
		/* linking: .libs/exefilename */
f6ccbe
		if (slbt_exec_link_create_executable(
f6ccbe
				dctx,ectx,
f6ccbe
				ectx->exefilename)) {
f6ccbe
			slbt_free_exec_ctx(actx);
f6ccbe
			return -1;
f6ccbe
		}
f6ccbe
	}
f6ccbe
2bd749
	/* no wrapper? */
5cc3b3
	if (!dot || strcmp(dot,".la")) {
2bd749
		slbt_free_exec_ctx(actx);
2bd749
		return 0;
2bd749
	}
2bd749
2bd749
	/* hey, yo, let's rap it up */
687567
	if (!(fout = fopen(output,"w"))) {
2bd749
		slbt_free_exec_ctx(actx);
2bd749
		return -1;
2bd749
	}
2bd749
4c76f7
	/* wrapper symlink */
6529aa
	if (slbt_create_symlink(
4c76f7
			dctx,ectx,
4c76f7
			output,
d1091b
			ectx->lafilename,
14ef98
			true))
14ef98
		ret = -1;
14ef98
	else
14ef98
		ret = fprintf(fout,
14ef98
			"# slibtool (pre-alpha) generated file\n\n");
4c76f7
2bd749
	/* all done */
2bd749
	fclose(fout);
2bd749
	slbt_free_exec_ctx(actx);
2bd749
2bd749
	return (ret > 0) ? 0 : -1;
2bd749
}