diff --git a/src/driver/mdso_unit_ctx.c b/src/driver/mdso_unit_ctx.c index 870e060..d955f7f 100644 --- a/src/driver/mdso_unit_ctx.c +++ b/src/driver/mdso_unit_ctx.c @@ -8,6 +8,8 @@ #include #include #include +#include +#include #include #include @@ -23,18 +25,62 @@ static int mdso_free_unit_ctx_impl(struct mdso_unit_ctx_impl * ctx, int status) return status; } +static FILE * mdso_stdin_to_tmp(void) +{ + FILE * ftmp; + char buf[4096]; + ssize_t nread; + int ret; + + if (!(ftmp = tmpfile())) + return 0; + + nread = read(0,buf,sizeof(buf)-1); + + while (nread) { + if (nread > 0) { + buf[nread] = '\0'; + ret = fputs(buf,ftmp); + } else + ret = (errno == EINTR) ? 0 : -1; + + if (ret < 0) { + fclose(ftmp); + return 0; + } + + nread = read(0,buf,sizeof(buf)-1); + } + + return ftmp; +} + int mdso_get_unit_ctx( const struct mdso_driver_ctx * dctx, const char * path, struct mdso_unit_ctx ** pctx) { struct mdso_unit_ctx_impl * ctx; + FILE * ftmp; + int fd; if (!dctx || !(ctx = calloc(sizeof(*ctx),1))) return -1; - if (mdso_map_input(-1,path,PROT_READ,&ctx->map)) + if (strcmp(path,"-")) + fd = -1; + else if (!(ftmp = mdso_stdin_to_tmp())) + return mdso_free_unit_ctx_impl(ctx,-1); + else if ((fd = dup(fileno(ftmp))) < 0) return mdso_free_unit_ctx_impl(ctx,-1); + else + fclose(ftmp); + + if (mdso_map_input(fd,path,PROT_READ,&ctx->map)) + return mdso_free_unit_ctx_impl(ctx,-1); + + if (fd > 0) + close(fd); memcpy(&ctx->cctx,dctx->cctx, sizeof(ctx->cctx));