| |
| |
| |
| |
| |
| |
| #ifndef _POSIX_C_SOURCE |
| #define _POSIX_C_SOURCE 200809L |
| #endif |
| |
| #include <fcntl.h> |
| #include <stdio.h> |
| #include <string.h> |
| #include <stdbool.h> |
| #include <fcntl.h> |
| #include <errno.h> |
| #include <sys/stat.h> |
| |
| #define ARGV_DRIVER |
| |
| #include <slibtool/slibtool.h> |
| #include "slibtool_driver_impl.h" |
| #include "slibtool_install_impl.h" |
| #include "slibtool_mapfile_impl.h" |
| #include "slibtool_readlink_impl.h" |
| #include "slibtool_spawn_impl.h" |
| #include "slibtool_symlink_impl.h" |
| #include "slibtool_errinfo_impl.h" |
| #include "argv/argv.h" |
| |
| static int slbt_install_usage( |
| int fdout, |
| const char * program, |
| const char * arg, |
| const struct argv_option ** optv, |
| struct argv_meta * meta) |
| { |
| char header[512]; |
| |
| snprintf(header,sizeof(header), |
| "Usage: %s --mode=install <install> [options] [SOURCE]... DEST\n" |
| "Options:\n", |
| program); |
| |
| argv_usage(fdout,header,optv,arg); |
| argv_free(meta); |
| |
| return SLBT_USAGE; |
| } |
| |
| static int slbt_exec_install_fail( |
| struct slbt_exec_ctx * actx, |
| struct argv_meta * meta, |
| int ret) |
| { |
| argv_free(meta); |
| slbt_free_exec_ctx(actx); |
| return ret; |
| } |
| |
| static int slbt_exec_install_init_dstdir( |
| const struct slbt_driver_ctx * dctx, |
| struct argv_entry * dest, |
| struct argv_entry * last, |
| char * dstdir) |
| { |
| struct stat st; |
| char * slash; |
| size_t len; |
| |
| (void)dctx; |
| |
| if (dest) |
| last = dest; |
| |
| |
| if ((size_t)snprintf(dstdir,PATH_MAX,"%s", |
| last->arg) >= PATH_MAX) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| len = strlen(dstdir); |
| |
| if (dstdir[--len] == '/') |
| dstdir[len] = 0; |
| |
| |
| if (dest) |
| return 0; |
| |
| |
| if (!(stat(dstdir,&st))) |
| if (S_ISDIR(st.st_mode)) |
| return 0; |
| |
| |
| if ((slash = strrchr(dstdir,'/'))) |
| *slash = 0; |
| |
| return 0; |
| } |
| |
| static int slbt_exec_install_import_libraries( |
| const struct slbt_driver_ctx * dctx, |
| struct slbt_exec_ctx * ectx, |
| char * srcdso, |
| char * dstdir) |
| { |
| char * host; |
| char * slash; |
| char * dot; |
| char * mark; |
| char srcbuf [PATH_MAX]; |
| char implib [PATH_MAX]; |
| char hosttag[PATH_MAX]; |
| char hostlnk[PATH_MAX]; |
| char major [128]; |
| char minor [128]; |
| char rev [128]; |
| |
| |
| if ((size_t)snprintf(srcbuf,sizeof(srcbuf),"%s", |
| srcdso) >= sizeof(srcbuf)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| if (!(slash = strrchr(srcbuf,'/'))) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| |
| |
| if ((size_t)snprintf(implib,sizeof(implib),"%s", |
| ++slash) >= sizeof(implib) |
| - strlen(dctx->cctx->settings.impsuffix)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| mark = srcbuf + strlen(srcbuf); |
| |
| |
| if (!(dot = strrchr(srcbuf,'.'))) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| else if ((size_t)(mark - dot) > sizeof(rev)) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_REV); |
| else { |
| strcpy(rev,dot); |
| *dot = 0; |
| } |
| |
| |
| if (!(dot = strrchr(srcbuf,'.'))) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| else if ((size_t)(mark - dot) > sizeof(minor)) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_REV); |
| else { |
| strcpy(minor,dot); |
| *dot = 0; |
| } |
| |
| |
| if (!(dot = strrchr(srcbuf,'.'))) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| else if ((size_t)(mark - dot) > sizeof(major)) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_REV); |
| else { |
| strcpy(major,dot); |
| *dot = 0; |
| } |
| |
| if (!(dot = strrchr(srcbuf,'.'))) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| |
| |
| if ((size_t)snprintf(hostlnk,sizeof(hostlnk),"%s.def.host", |
| srcbuf) >= sizeof(hostlnk)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| if (slbt_readlink(hostlnk,hosttag,sizeof(hosttag))) |
| return SLBT_SYSTEM_ERROR(dctx); |
| |
| |
| if (!(host = strrchr(hosttag,'.'))) |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| else |
| host++; |
| |
| |
| if (slbt_set_alternate_host(dctx,host,host)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| |
| sprintf(dot,"%s%s%s%s", |
| major,minor,rev, |
| dctx->cctx->asettings.impsuffix); |
| |
| |
| if (slbt_copy_file(dctx,ectx,srcbuf,dstdir)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| |
| sprintf(dot,"%s%s", |
| major, |
| dctx->cctx->asettings.impsuffix); |
| |
| |
| if (slbt_copy_file(dctx,ectx,srcbuf,dstdir)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| |
| strcpy(implib,slash); |
| strcpy(dot,dctx->cctx->asettings.impsuffix); |
| |
| if ((size_t)snprintf(hostlnk,sizeof(hostlnk),"%s/%s", |
| dstdir,slash) >= sizeof(hostlnk)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| if (slbt_create_symlink( |
| dctx,ectx, |
| implib, |
| hostlnk, |
| false)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| return 0; |
| } |
| |
| static int slbt_exec_install_library_wrapper( |
| const struct slbt_driver_ctx * dctx, |
| struct slbt_exec_ctx * ectx, |
| struct argv_entry * entry, |
| char * dstdir) |
| { |
| int fdcwd; |
| int fddst; |
| size_t buflen; |
| const char * base; |
| char * srcline; |
| char * dstline; |
| char clainame[PATH_MAX]; |
| char instname[PATH_MAX]; |
| char cfgbuf [PATH_MAX]; |
| struct slbt_map_info * mapinfo; |
| |
| |
| if ((base = strrchr(entry->arg,'/'))) |
| base++; |
| else |
| base = entry->arg; |
| |
| |
| if ((size_t)snprintf(instname,sizeof(instname),"%s/%s", |
| dstdir,base) >= sizeof(instname)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| if ((size_t)snprintf(clainame,sizeof(clainame),"%s.slibtool.install", |
| entry->arg) >= sizeof(clainame)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| fdcwd = slbt_driver_fdcwd(dctx); |
| |
| |
| if ((fddst = openat(fdcwd,clainame,O_RDWR|O_CREAT|O_TRUNC,0644)) < 0) |
| return SLBT_SYSTEM_ERROR(dctx); |
| |
| |
| if (!(mapinfo = slbt_map_file(fdcwd,entry->arg,SLBT_MAP_INPUT))) { |
| close(fddst); |
| return SLBT_SYSTEM_ERROR(dctx); |
| } |
| |
| |
| if (mapinfo->size < sizeof(cfgbuf)) { |
| buflen = sizeof(cfgbuf); |
| srcline = cfgbuf; |
| } else { |
| buflen = mapinfo->size; |
| srcline = malloc(++buflen); |
| } |
| |
| if (!srcline) { |
| close(fddst); |
| slbt_unmap_file(mapinfo); |
| return SLBT_SYSTEM_ERROR(dctx); |
| } |
| |
| |
| while (mapinfo->mark < mapinfo->cap) { |
| if (slbt_mapped_readline(dctx,mapinfo,srcline,buflen) < 0) { |
| close(fddst); |
| slbt_unmap_file(mapinfo); |
| return SLBT_NESTED_ERROR(dctx); |
| } |
| |
| dstline = strcmp(srcline,"installed=no\n") |
| ? srcline |
| : "installed=yes\n"; |
| |
| if (slbt_dprintf(fddst,"%s",dstline) < 0) { |
| close(fddst); |
| slbt_unmap_file(mapinfo); |
| return SLBT_SYSTEM_ERROR(dctx); |
| } |
| } |
| |
| if (srcline != cfgbuf) |
| free(srcline); |
| |
| |
| close(fddst); |
| slbt_unmap_file(mapinfo); |
| |
| |
| if (slbt_copy_file(dctx,ectx,clainame,instname)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| return 0; |
| } |
| |
| static int slbt_exec_install_entry( |
| const struct slbt_driver_ctx * dctx, |
| struct slbt_exec_ctx * ectx, |
| struct argv_entry * entry, |
| struct argv_entry * last, |
| struct argv_entry * dest, |
| char * dstdir, |
| char ** src, |
| char ** dst) |
| { |
| int ret; |
| char * dot; |
| char * base; |
| char * slash; |
| char target [PATH_MAX]; |
| char srcfile [PATH_MAX]; |
| char dstfile [PATH_MAX]; |
| char slnkname[PATH_MAX]; |
| char dlnkname[PATH_MAX]; |
| char lasource[PATH_MAX - 8]; |
| bool fexe = false; |
| bool fpe; |
| bool frelease; |
| struct stat st; |
| |
| |
| if ((size_t)snprintf(slnkname,sizeof(slnkname),"%s.exe.wrapper", |
| entry->arg) >= sizeof(slnkname)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| fexe = stat(slnkname,&st) |
| ? false |
| : true; |
| |
| dot = strrchr(entry->arg,'.'); |
| |
| |
| if (!fexe && dot && !strcmp(dot,".lai")) |
| dot[3] = 0; |
| |
| |
| if (strlen(entry->arg) + strlen(".libs/") >= (PATH_MAX-1)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| strcpy(lasource,entry->arg); |
| |
| if ((slash = strrchr(lasource,'/'))) { |
| *slash++ = 0; |
| sprintf(srcfile,"%s/.libs/%s",lasource,slash); |
| } else |
| sprintf(srcfile,".libs/%s",lasource); |
| |
| |
| if (fexe || !dot || strcmp(dot,".la")) { |
| *src = fexe ? srcfile : (char *)entry->arg; |
| *dst = dest ? 0 : (char *)last->arg; |
| |
| if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT)) |
| if (slbt_output_install(dctx,ectx)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| return (((ret = slbt_spawn(ectx,true)) < 0) || ectx->exitcode) |
| ? SLBT_SPAWN_ERROR(dctx) : 0; |
| } |
| |
| |
| if (dctx->cctx->drvflags & SLBT_DRIVER_LEGABITS) |
| if (slbt_exec_install_library_wrapper(dctx,ectx,entry,dstdir)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| |
| if ((*dst = dest ? 0 : (char *)last->arg)) |
| if ((dot = strrchr(last->arg,'.'))) |
| if (!(strcmp(dot,".la"))) |
| *dst = dstdir; |
| |
| |
| dot = strrchr(srcfile,'.'); |
| strcpy(dot,dctx->cctx->settings.arsuffix); |
| |
| if (!(dctx->cctx->drvflags & SLBT_DRIVER_DISABLE_STATIC)) |
| if (slbt_copy_file(dctx,ectx, |
| srcfile, |
| dest ? (char *)dest->arg : *dst)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| |
| strcpy(slnkname,srcfile); |
| dot = strrchr(slnkname,'.'); |
| |
| |
| sprintf(dot,"%s.release",dctx->cctx->settings.dsosuffix); |
| frelease = stat(slnkname,&st) ? false : true; |
| |
| |
| strcpy(dot,dctx->cctx->settings.dsosuffix); |
| |
| |
| if ((size_t)snprintf(dstfile,sizeof(dstfile),"%s.def", |
| slnkname) >= sizeof(dstfile)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| fpe = stat(dstfile,&st) ? false : true; |
| |
| |
| if ((base = strrchr(slnkname,'/'))) |
| base++; |
| else |
| base = slnkname; |
| |
| |
| if (slbt_readlink(slnkname,target,sizeof(target)) < 0) { |
| |
| if (slbt_symlink_is_a_placeholder(slnkname)) |
| return 0; |
| |
| |
| if (stat(slnkname,&st)) |
| return SLBT_SYSTEM_ERROR(dctx); |
| |
| |
| if ((size_t)snprintf(dstfile,sizeof(dstfile),"%s/%s", |
| dstdir,base) >= sizeof(dstfile)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| *src = slnkname; |
| *dst = dest ? 0 : dstfile; |
| |
| if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT)) |
| if (slbt_output_install(dctx,ectx)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| if (((ret = slbt_spawn(ectx,true)) < 0) || ectx->exitcode) |
| return SLBT_SPAWN_ERROR(dctx); |
| |
| return 0; |
| } |
| |
| |
| slash = strrchr(srcfile,'/'); |
| strcpy(++slash,target); |
| |
| |
| if (!dest) |
| if ((size_t)snprintf(dstfile,sizeof(dstfile),"%s/%s", |
| dstdir,target) >= sizeof(dstfile)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| *src = srcfile; |
| *dst = dest ? 0 : dstfile; |
| |
| if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT)) |
| if (slbt_output_install(dctx,ectx)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| if (((ret = slbt_spawn(ectx,true)) < 0) || ectx->exitcode) |
| return SLBT_SPAWN_ERROR(dctx); |
| |
| |
| if ((size_t)snprintf(dlnkname,sizeof(dlnkname),"%s/%s", |
| dstdir,base) >= sizeof(dlnkname)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| |
| if (slbt_create_symlink( |
| dctx,ectx, |
| target,dlnkname, |
| false)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| if (frelease) |
| return 0; |
| |
| |
| strcpy(slnkname,target); |
| |
| if ((dot = strrchr(slnkname,'.'))) |
| *dot = 0; |
| else |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| |
| if ((dot = strrchr(slnkname,'.'))) |
| *dot = 0; |
| else |
| return SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FLOW); |
| |
| |
| if ((size_t)snprintf(dlnkname,sizeof(dlnkname),"%s/%s", |
| dstdir,slnkname) >= sizeof(dlnkname)) |
| return SLBT_BUFFER_ERROR(dctx); |
| |
| if (fpe) { |
| |
| if (slbt_copy_file( |
| dctx,ectx, |
| srcfile, |
| dlnkname)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| |
| if (slbt_exec_install_import_libraries( |
| dctx,ectx, |
| srcfile, |
| dstdir)) |
| return SLBT_NESTED_ERROR(dctx); |
| } else { |
| |
| if (slbt_create_symlink( |
| dctx,ectx, |
| target,dlnkname, |
| false)) |
| return SLBT_NESTED_ERROR(dctx); |
| } |
| |
| return 0; |
| } |
| |
| int slbt_exec_install( |
| const struct slbt_driver_ctx * dctx, |
| struct slbt_exec_ctx * ectx) |
| { |
| int ret; |
| int fdout; |
| char ** argv; |
| char ** iargv; |
| char ** src; |
| char ** dst; |
| struct slbt_exec_ctx * actx; |
| struct argv_meta * meta; |
| struct argv_entry * entry; |
| struct argv_entry * copy; |
| struct argv_entry * dest; |
| struct argv_entry * last; |
| const struct argv_option * optv[SLBT_OPTV_ELEMENTS]; |
| char dstdir[PATH_MAX]; |
| |
| |
| if (dctx->cctx->drvflags & SLBT_DRIVER_DRY_RUN) |
| return 0; |
| |
| |
| if (ectx) |
| actx = 0; |
| else if ((ret = slbt_get_exec_ctx(dctx,&ectx))) |
| return ret; |
| else |
| actx = ectx; |
| |
| |
| slbt_reset_arguments(ectx); |
| slbt_disable_placeholders(ectx); |
| iargv = ectx->cargv; |
| fdout = slbt_driver_fdout(dctx); |
| |
| |
| if (!(strcmp(iargv[0],"/bin/sh")) || !strcmp(iargv[0],"/bin/bash")) |
| iargv++; |
| |
| |
| argv_optv_init(slbt_install_options,optv); |
| |
| if (!iargv[1] && (dctx->cctx->drvflags & SLBT_DRIVER_VERBOSITY_USAGE)) |
| return slbt_install_usage( |
| fdout, |
| dctx->program, |
| 0,optv,0); |
| |
| |
| if (!(meta = argv_get( |
| iargv,optv, |
| dctx->cctx->drvflags & SLBT_DRIVER_VERBOSITY_ERRORS |
| ? ARGV_VERBOSITY_ERRORS |
| : ARGV_VERBOSITY_NONE, |
| fdout))) |
| return slbt_exec_install_fail( |
| actx,meta, |
| SLBT_CUSTOM_ERROR(dctx,SLBT_ERR_INSTALL_FAIL)); |
| |
| |
| argv = ectx->altv; |
| copy = meta->entries; |
| dest = 0; |
| last = 0; |
| |
| *argv++ = iargv[0]; |
| |
| for (entry=meta->entries; entry->fopt || entry->arg; entry++) { |
| if (entry->fopt) { |
| switch (entry->tag) { |
| case TAG_INSTALL_COPY: |
| *argv++ = "-c"; |
| copy = entry; |
| break; |
| |
| case TAG_INSTALL_FORCE: |
| *argv++ = "-f"; |
| break; |
| |
| case TAG_INSTALL_MKDIR: |
| *argv++ = "-d"; |
| copy = 0; |
| break; |
| |
| case TAG_INSTALL_TARGET_MKDIR: |
| *argv++ = "-D"; |
| copy = 0; |
| break; |
| |
| case TAG_INSTALL_STRIP: |
| *argv++ = "-s"; |
| break; |
| |
| case TAG_INSTALL_PRESERVE: |
| *argv++ = "-p"; |
| break; |
| |
| case TAG_INSTALL_USER: |
| *argv++ = "-o"; |
| break; |
| |
| case TAG_INSTALL_GROUP: |
| *argv++ = "-g"; |
| break; |
| |
| case TAG_INSTALL_MODE: |
| *argv++ = "-m"; |
| break; |
| |
| case TAG_INSTALL_DSTDIR: |
| *argv++ = "-t"; |
| dest = entry; |
| break; |
| } |
| |
| if (entry->fval) |
| *argv++ = (char *)entry->arg; |
| } else |
| last = entry; |
| } |
| |
| |
| if (copy) { |
| |
| ectx->argv = ectx->altv; |
| ectx->program = ectx->altv[0]; |
| |
| |
| src = argv++; |
| dst = argv++; |
| |
| |
| if (slbt_exec_install_init_dstdir(dctx,dest,last,dstdir)) |
| return slbt_exec_install_fail( |
| actx,meta, |
| SLBT_NESTED_ERROR(dctx)); |
| |
| |
| for (entry=meta->entries; entry->fopt || entry->arg; entry++) |
| if (!entry->fopt && (dest || (entry != last))) |
| if (slbt_exec_install_entry( |
| dctx,ectx, |
| entry,last, |
| dest,dstdir, |
| src,dst)) |
| return slbt_exec_install_fail( |
| actx,meta, |
| SLBT_NESTED_ERROR(dctx)); |
| } else { |
| |
| ectx->argv = ectx->cargv; |
| ectx->program = ectx->cargv[0]; |
| |
| |
| if (!(dctx->cctx->drvflags & SLBT_DRIVER_SILENT)) |
| if (slbt_output_install(dctx,ectx)) |
| return SLBT_NESTED_ERROR(dctx); |
| |
| if (((ret = slbt_spawn(ectx,true)) < 0) || ectx->exitcode) |
| return slbt_exec_install_fail( |
| actx,meta, |
| SLBT_SPAWN_ERROR(dctx)); |
| } |
| |
| argv_free(meta); |
| slbt_free_exec_ctx(actx); |
| |
| return 0; |
| } |