firasuke / cross / slibtool

Forked from cross/slibtool 4 months ago
Clone

Blame src/logic/linkcmd/slbt_linkcmd_archive.c

3bfc81
/*******************************************************************/
eac61a
/*  slibtool: a strong libtool implementation, written in C        */
49181b
/*  Copyright (C) 2016--2024  SysDeer Technologies, LLC            */
3bfc81
/*  Released under the Standard MIT License; see COPYING.SLIBTOOL. */
3bfc81
/*******************************************************************/
3bfc81
3bfc81
#include <stdlib.h>
3bfc81
#include <stdio.h>
3bfc81
#include <string.h>
3bfc81
#include <fcntl.h>
3bfc81
#include <errno.h>
3bfc81
#include <sys/stat.h>
3bfc81
3bfc81
#include <slibtool/slibtool.h>
3bfc81
#include "slibtool_driver_impl.h"
3bfc81
#include "slibtool_errinfo_impl.h"
3bfc81
#include "slibtool_linkcmd_impl.h"
3bfc81
#include "slibtool_mapfile_impl.h"
3bfc81
#include "slibtool_metafile_impl.h"
3bfc81
#include "slibtool_snprintf_impl.h"
3bfc81
#include "slibtool_symlink_impl.h"
3bfc81
#include "slibtool_spawn_impl.h"
4b56de
#include "slibtool_visibility_impl.h"
3bfc81
3bfc81
static int slbt_exec_link_remove_file(
3bfc81
	const struct slbt_driver_ctx *	dctx,
3bfc81
	struct slbt_exec_ctx *		ectx,
3bfc81
	const char *			target)
3bfc81
{
3bfc81
	int fdcwd;
93e38b
	char * mark;
93e38b
	char * sbuf;
3bfc81
3bfc81
	/* fdcwd */
3bfc81
	fdcwd = slbt_driver_fdcwd(dctx);
3bfc81
3bfc81
	/* remove target (if any) */
93e38b
	if (unlinkat(fdcwd,target,0) && (errno != ENOENT))
93e38b
		return SLBT_SYSTEM_ERROR(dctx,0);
93e38b
93e38b
	/* remove a previous .disabled placeholder */
93e38b
	sbuf  = (slbt_get_exec_ictx(ectx))->sbuf;
93e38b
	mark  = sbuf;
93e38b
	mark += sprintf(mark,"%s",target);
93e38b
	strcpy(mark,".disabled");
93e38b
93e38b
	if (unlinkat(fdcwd,sbuf,0) && (errno != ENOENT))
93e38b
		return SLBT_SYSTEM_ERROR(dctx,0);
3bfc81
93e38b
	return 0;
3bfc81
}
3bfc81
15782c
15782c
static bool slbt_archive_is_convenience_library(int fdcwd, const char * arpath)
15782c
{
15782c
	int     fd;
15782c
	char    laipath[PATH_MAX];
15782c
	char *  dot;
15782c
15782c
	strcpy(laipath,arpath);
15782c
	dot = strrchr(laipath,'.');
15782c
	strcpy(dot,".lai");
15782c
15782c
	if ((fd = openat(fdcwd,laipath,O_RDONLY,0)) >= 0) {
15782c
		close(fd);
15782c
		return false;
15782c
	}
15782c
15782c
	return true;
15782c
}
15782c
15782c
4b56de
slbt_hidden int slbt_exec_link_create_archive(
3bfc81
	const struct slbt_driver_ctx *	dctx,
3bfc81
	struct slbt_exec_ctx *		ectx,
3bfc81
	const char *			arfilename,
7c3e61
	bool				fpic,
7c3e61
	bool                            fdep)
3bfc81
{
3bfc81
	int		fdcwd;
a1c68a
	char **         argv;
3bfc81
	char ** 	aarg;
3bfc81
	char ** 	parg;
3bfc81
	char		program[PATH_MAX];
3bfc81
	char		output [PATH_MAX];
7a5788
	char		namebuf[PATH_MAX];
3bfc81
1769a4
	/* dlopen, dlpreopen: object compilation (derived from dynamic linking) */
1769a4
	if (ectx->dlopenobj) {
1769a4
		slbt_ectx_reset_arguments(ectx);
1769a4
		slbt_reset_placeholders(ectx);
1769a4
7a5788
		sprintf(namebuf,"%s%s",ectx->ldirname,"@ARDLOPEN@");
7a5788
1769a4
		if (slbt_exec_link_create_library(
1769a4
				dctx,ectx,
7a5788
				namebuf,namebuf,namebuf,
39ce39
				true,fpic) < 0)
1769a4
			return SLBT_NESTED_ERROR(dctx);
1769a4
	}
1769a4
1769a4
	/* restore initial state */
f3d47a
	slbt_ectx_reset_arguments(ectx);
3bfc81
3bfc81
	/* placeholders */
3bfc81
	slbt_reset_placeholders(ectx);
3bfc81
3bfc81
	/* output */
3bfc81
	if (slbt_snprintf(output,sizeof(output),
3bfc81
			"%s",arfilename) < 0)
3bfc81
		return SLBT_BUFFER_ERROR(dctx);
3bfc81
a1c68a
	/* tool-specific argument vector */
a1c68a
	argv = (slbt_get_driver_ictx(dctx))->host.ar_argv;
3bfc81
3bfc81
	/* fdcwd */
3bfc81
	fdcwd   = slbt_driver_fdcwd(dctx);
3bfc81
3bfc81
	/* input argument adjustment */
3bfc81
	aarg    = ectx->altv;
a1c68a
a1c68a
	if ((parg = argv)) {
a1c68a
		for (; *parg; )
a1c68a
			*aarg++ = *parg++;
a1c68a
	} else {
a1c68a
		*aarg++ = program;
a1c68a
	}
a1c68a
3bfc81
	*aarg++ = "-crs";
3bfc81
	*aarg++ = output;
3bfc81
3bfc81
	for (parg=ectx->cargv; *parg; parg++)
3bfc81
		if (slbt_adjust_object_argument(*parg,fpic,!fpic,fdcwd))
3bfc81
			*aarg++ = *parg;
3bfc81
1769a4
	if (ectx->dlopenobj)
1769a4
		*aarg++ = ectx->dlopenobj;
1769a4
3bfc81
	*aarg = 0;
3bfc81
	ectx->argv = ectx->altv;
3bfc81
1769a4
	/* ar program */
1769a4
	if (argv) {
1769a4
		ectx->program = argv[0];
1769a4
	} else {
1769a4
		if (slbt_snprintf(program,sizeof(program),
1769a4
				"%s",dctx->cctx->host.ar) < 0)
1769a4
			return SLBT_BUFFER_ERROR(dctx);
1769a4
1769a4
		ectx->program = program;
1769a4
	}
1769a4
3bfc81
	/* step output */
3bfc81
	if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT))
33a569
		if (slbt_output_link(ectx))
3bfc81
			return SLBT_NESTED_ERROR(dctx);
3bfc81
3bfc81
	/* remove old archive as needed */
3bfc81
	if (slbt_exec_link_remove_file(dctx,ectx,output))
3bfc81
		return SLBT_NESTED_ERROR(dctx);
3bfc81
3bfc81
	/* .deps */
7c3e61
	if (fdep)
7c3e61
		if (slbt_exec_link_create_dep_file(
7c3e61
				dctx,ectx,ectx->cargv,
7c3e61
				arfilename,true))
7c3e61
			return SLBT_NESTED_ERROR(dctx);
3bfc81
3bfc81
	/* ar spawn */
3bfc81
	if ((slbt_spawn(ectx,true) < 0) && (ectx->pid < 0)) {
3bfc81
		return SLBT_SPAWN_ERROR(dctx);
3bfc81
3bfc81
	} else if (ectx->exitcode) {
3bfc81
		return SLBT_CUSTOM_ERROR(
3bfc81
			dctx,
3bfc81
			SLBT_ERR_AR_ERROR);
3bfc81
	}
3bfc81
3bfc81
	/* input objects associated with .la archives */
3bfc81
	for (parg=ectx->cargv; *parg; parg++)
d054ee
		if (slbt_adjust_wrapper_argument(
d054ee
				*parg,true,
d054ee
				dctx->cctx->settings.arsuffix))
15782c
			if (slbt_archive_is_convenience_library(fdcwd,*parg))
15782c
				if (slbt_util_import_archive(ectx,output,*parg))
15782c
					return SLBT_NESTED_ERROR(dctx);
3bfc81
	return 0;
3bfc81
}