| |
| |
| |
| |
| |
| |
| #include <stdio.h> |
| #include <string.h> |
| #include <stdbool.h> |
| #include <unistd.h> |
| |
| #include "slibtool_errinfo_impl.h" |
| #include "slibtool_symlink_impl.h" |
| #include "slibtool_readlink_impl.h" |
| |
| #define SLBT_DEV_NULL_FLAGS (SLBT_DRIVER_ALL_STATIC \ |
| | SLBT_DRIVER_DISABLE_SHARED \ |
| | SLBT_DRIVER_DISABLE_STATIC) |
| |
| int slbt_create_symlink( |
| const struct slbt_driver_ctx * dctx, |
| struct slbt_exec_ctx * ectx, |
| const char * target, |
| const char * lnkname, |
| bool flawrapper) |
| { |
| char ** oargv; |
| const char * slash; |
| char * ln[5]; |
| char * dotdot; |
| char tmplnk [PATH_MAX]; |
| char lnkarg [PATH_MAX]; |
| char alnkarg[PATH_MAX]; |
| char atarget[PATH_MAX]; |
| char * suffix = 0; |
| |
| |
| if ((dctx->cctx->drvflags & SLBT_DEV_NULL_FLAGS) |
| && !strcmp(target,"/dev/null")) { |
| slash = target; |
| suffix = ".disabled"; |
| |
| |
| } else if ((slash = strrchr(target,'/'))) { |
| slash++; |
| |
| |
| } else { |
| slash = target; |
| } |
| |
| |
| dotdot = flawrapper ? "../" : ""; |
| |
| |
| if ((size_t)snprintf(atarget,sizeof(atarget),"%s%s", |
| dotdot,slash) >= sizeof(atarget)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| if ((size_t)snprintf(tmplnk,sizeof(tmplnk),"%s.symlink.tmp", |
| lnkname) >= sizeof(tmplnk)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| if (suffix) { |
| sprintf(alnkarg,"%s%s",lnkname,suffix); |
| lnkname = alnkarg; |
| } |
| |
| |
| strcpy(lnkarg,lnkname); |
| |
| |
| ln[0] = "ln"; |
| ln[1] = "-s"; |
| ln[2] = atarget; |
| ln[3] = lnkarg; |
| ln[4] = 0; |
| |
| oargv = ectx->argv; |
| ectx->argv = ln; |
| |
| |
| if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT)) { |
| if (dctx->cctx->mode == SLBT_MODE_LINK) { |
| if (slbt_output_link(dctx,ectx)) { |
| ectx->argv = oargv; |
| return SLBT_NESTED_ERROR(dctx); |
| } |
| } else { |
| if (slbt_output_install(dctx,ectx)) { |
| ectx->argv = oargv; |
| return SLBT_NESTED_ERROR(dctx); |
| } |
| } |
| } |
| |
| |
| ectx->argv = oargv; |
| |
| |
| if (symlink(atarget,tmplnk)) |
| return SLBT_SYSTEM_ERROR(dctx); |
| |
| return rename(tmplnk,lnkname) |
| ? SLBT_SYSTEM_ERROR(dctx) |
| : 0; |
| } |
| |
| int slbt_symlink_is_a_placeholder(char * lnkpath) |
| { |
| size_t len; |
| char slink [PATH_MAX]; |
| char target[PATH_MAX]; |
| const char suffix[] = ".disabled"; |
| |
| if ((sizeof(slink)-sizeof(suffix)) < (len=strlen(lnkpath))) |
| return 0; |
| |
| memcpy(slink,lnkpath,len); |
| memcpy(&slink[len],suffix,sizeof(suffix)); |
| |
| return (!slbt_readlink(slink,target,sizeof(target))) |
| && (!strcmp(target,"/dev/null")); |
| } |