diff options
author | midipix <writeonce@midipix.org> | 2024-03-25 05:38:35 +0000 |
---|---|---|
committer | midipix <writeonce@midipix.org> | 2024-03-25 06:42:04 +0000 |
commit | 44fa06c39b5be9060a268a9b308ae4217d45da4f (patch) | |
tree | 4e8683cc3732e668f3d22b67848a99329e103c7a /src/internal/slibtool_symlink_impl.c | |
parent | 01d76383b6fe9b44cc3dd151f2757dcd503932fc (diff) | |
download | slibtool-44fa06c39b5be9060a268a9b308ae4217d45da4f.tar.bz2 slibtool-44fa06c39b5be9060a268a9b308ae4217d45da4f.tar.xz |
internals: added slbt_create_symlink_ex() [specify arbitrary destination dir].
Diffstat (limited to 'src/internal/slibtool_symlink_impl.c')
-rw-r--r-- | src/internal/slibtool_symlink_impl.c | 45 |
1 files changed, 35 insertions, 10 deletions
diff --git a/src/internal/slibtool_symlink_impl.c b/src/internal/slibtool_symlink_impl.c index 7a1b997..bb117e2 100644 --- a/src/internal/slibtool_symlink_impl.c +++ b/src/internal/slibtool_symlink_impl.c @@ -13,6 +13,7 @@ #include "slibtool_errinfo_impl.h" #include "slibtool_symlink_impl.h" #include "slibtool_readlink_impl.h" +#include "slibtool_realpath_impl.h" #include "slibtool_snprintf_impl.h" #include "slibtool_visibility_impl.h" @@ -20,22 +21,24 @@ | SLBT_DRIVER_DISABLE_SHARED \ | SLBT_DRIVER_DISABLE_STATIC) -slbt_hidden int slbt_create_symlink( +slbt_hidden int slbt_create_symlink_ex( const struct slbt_driver_ctx * dctx, struct slbt_exec_ctx * ectx, + int fddst, const char * target, const char * lnkname, uint32_t options) { - int fdcwd; int fliteral; int fwrapper; int fdevnull; + size_t slen; char ** oargv; const char * slash; char * ln[5]; char * dot; char * dotdot; + char * mark; char tmplnk [PATH_MAX]; char lnkarg [PATH_MAX]; char alnkarg[PATH_MAX]; @@ -79,19 +82,16 @@ slbt_hidden int slbt_create_symlink( lnkname) <0) return SLBT_BUFFER_ERROR(dctx); - /* fdcwd */ - fdcwd = slbt_driver_fdcwd(dctx); - /* placeholder? */ if (fdevnull) { - if (unlinkat(fdcwd,lnkname,0) && (errno != ENOENT)) + if (unlinkat(fddst,lnkname,0) && (errno != ENOENT)) return SLBT_SYSTEM_ERROR(dctx,0); if ((dot = strrchr(lnkname,'.'))) { if (!strcmp(dot,dctx->cctx->settings.dsosuffix)) { strcpy(dot,".expsyms.a"); - if (unlinkat(fdcwd,lnkname,0) && (errno != ENOENT)) + if (unlinkat(fddst,lnkname,0) && (errno != ENOENT)) return SLBT_SYSTEM_ERROR(dctx,0); strcpy(dot,dctx->cctx->settings.dsosuffix); @@ -105,7 +105,19 @@ slbt_hidden int slbt_create_symlink( } /* lnkarg */ - strcpy(lnkarg,lnkname); + if (fddst == slbt_driver_fdcwd(dctx)) { + strcpy(lnkarg,lnkname); + } else { + if (slbt_realpath(fddst,".",0,lnkarg,sizeof(lnkarg)) < 0) + return SLBT_BUFFER_ERROR(dctx); + + if ((slen = strlen(lnkarg)) + strlen(lnkname) + 1 >= PATH_MAX) + return SLBT_BUFFER_ERROR(dctx); + + mark = &lnkarg[slen]; + mark[0] = '/'; + strcpy(++mark,lnkname); + } /* ln argv (fake) */ ln[0] = "ln"; @@ -136,14 +148,27 @@ slbt_hidden int slbt_create_symlink( ectx->argv = oargv; /* create symlink */ - if (symlinkat(atarget,fdcwd,tmplnk)) + if (symlinkat(atarget,fddst,tmplnk)) return SLBT_SYSTEM_ERROR(dctx,tmplnk); - return renameat(fdcwd,tmplnk,fdcwd,lnkname) + return renameat(fddst,tmplnk,fddst,lnkname) ? SLBT_SYSTEM_ERROR(dctx,lnkname) : 0; } +slbt_hidden int slbt_create_symlink( + const struct slbt_driver_ctx * dctx, + struct slbt_exec_ctx * ectx, + const char * target, + const char * lnkname, + uint32_t options) +{ + return slbt_create_symlink_ex( + dctx,ectx, + slbt_driver_fdcwd(dctx), + target,lnkname,options); +} + slbt_hidden int slbt_symlink_is_a_placeholder(int fdcwd, const char * lnkpath) { size_t len; |