|
|
383aa6 |
/**********************************************************/
|
|
|
383aa6 |
/* apimagic: cparser-based API normalization utility */
|
|
|
383aa6 |
/* Copyright (C) 2015--2016 Z. Gilboa */
|
|
|
383aa6 |
/* Released under GPLv2 and GPLv3; see COPYING.APIMAGIC. */
|
|
|
383aa6 |
/**********************************************************/
|
|
|
383aa6 |
|
|
|
383aa6 |
#include <stdint.h>
|
|
|
383aa6 |
#include <stddef.h>
|
|
|
383aa6 |
#include <stdlib.h>
|
|
|
383aa6 |
#include <string.h>
|
|
|
389acb |
#include <unistd.h>
|
|
|
389acb |
#include <errno.h>
|
|
|
383aa6 |
#include <sys/mman.h>
|
|
|
383aa6 |
|
|
|
43c8c9 |
|
|
|
43c8c9 |
#include <cparser/driver/driver.h>
|
|
|
43c8c9 |
#include <cparser/driver/c_driver.h>
|
|
|
43c8c9 |
|
|
|
383aa6 |
#include <apimagic/apimagic.h>
|
|
|
383aa6 |
#include "apimagic_driver_impl.h"
|
|
|
4671ad |
#include "apimagic_errinfo_impl.h"
|
|
|
383aa6 |
|
|
|
4671ad |
static int amgc_free_unit_ctx_impl(struct amgc_unit_ctx_impl * ctx, int ret)
|
|
|
383aa6 |
{
|
|
|
383aa6 |
if (ctx) {
|
|
|
7a622c |
if (ctx->entities)
|
|
|
7a622c |
amgc_free_unit_entities(ctx->entities);
|
|
|
7a622c |
|
|
|
383aa6 |
amgc_unmap_input(&ctx->map);
|
|
|
383aa6 |
free(ctx);
|
|
|
383aa6 |
}
|
|
|
383aa6 |
|
|
|
4671ad |
return ret;
|
|
|
383aa6 |
}
|
|
|
383aa6 |
|
|
|
389acb |
static FILE * amgc_stdin_to_tmp(const struct amgc_driver_ctx * dctx)
|
|
|
389acb |
{
|
|
|
389acb |
struct amgc_driver_ctx_impl * ictx;
|
|
|
389acb |
uintptr_t addr;
|
|
|
389acb |
int fdtmp;
|
|
|
389acb |
|
|
|
389acb |
FILE * ftmp;
|
|
|
389acb |
char buf[4096];
|
|
|
389acb |
ssize_t nread;
|
|
|
389acb |
int ret;
|
|
|
389acb |
|
|
|
389acb |
addr = (uintptr_t)dctx - offsetof(struct amgc_driver_ctx_impl,ctx);
|
|
|
389acb |
ictx = (struct amgc_driver_ctx_impl *)addr;
|
|
|
389acb |
|
|
|
389acb |
if (ictx->fdtmpin >= 0) {
|
|
|
389acb |
if ((fdtmp = dup(ictx->fdtmpin)) < 0)
|
|
|
389acb |
return 0;
|
|
|
389acb |
|
|
|
389acb |
if (!(ftmp = fdopen(fdtmp,"r")))
|
|
|
389acb |
close(fdtmp);
|
|
|
389acb |
|
|
|
389acb |
return ftmp;
|
|
|
389acb |
}
|
|
|
389acb |
|
|
|
389acb |
if (!(ftmp = tmpfile()))
|
|
|
389acb |
return 0;
|
|
|
389acb |
|
|
|
389acb |
if ((ictx->fdtmpin = dup(fileno(ftmp))) < 0) {
|
|
|
389acb |
fclose(ftmp);
|
|
|
389acb |
return 0;
|
|
|
389acb |
}
|
|
|
389acb |
|
|
|
389acb |
nread = read(0,buf,sizeof(buf)-1);
|
|
|
389acb |
|
|
|
389acb |
while (nread) {
|
|
|
389acb |
if (nread > 0) {
|
|
|
389acb |
buf[nread] = '\0';
|
|
|
389acb |
ret = fputs(buf,ftmp);
|
|
|
389acb |
} else
|
|
|
389acb |
ret = (errno == EINTR) ? 0 : -1;
|
|
|
389acb |
|
|
|
389acb |
if (ret < 0) {
|
|
|
389acb |
fclose(ftmp);
|
|
|
389acb |
return 0;
|
|
|
389acb |
}
|
|
|
389acb |
|
|
|
389acb |
nread = read(0,buf,sizeof(buf)-1);
|
|
|
389acb |
}
|
|
|
389acb |
|
|
|
389acb |
return ftmp;
|
|
|
389acb |
}
|
|
|
389acb |
|
|
|
43c8c9 |
static bool amgc_cparser_no_op(
|
|
|
43c8c9 |
compilation_env_t * env,
|
|
|
43c8c9 |
compilation_unit_t * unit)
|
|
|
43c8c9 |
{
|
|
|
3a51d2 |
(void)env;
|
|
|
3a51d2 |
(void)unit;
|
|
|
43c8c9 |
return true;
|
|
|
43c8c9 |
}
|
|
|
43c8c9 |
|
|
|
43c8c9 |
static void amgc_init_cparser_unit(void)
|
|
|
43c8c9 |
{
|
|
|
43c8c9 |
set_default_handlers();
|
|
|
43c8c9 |
|
|
|
43c8c9 |
/* parse only */
|
|
|
43c8c9 |
set_unit_handler(COMPILATION_UNIT_AST,
|
|
|
43c8c9 |
build_firm_ir,true);
|
|
|
43c8c9 |
|
|
|
43c8c9 |
/* assign no-op handler */
|
|
|
43c8c9 |
set_unit_handler(COMPILATION_UNIT_PREPROCESSED_ASSEMBLER,
|
|
|
43c8c9 |
amgc_cparser_no_op,true);
|
|
|
43c8c9 |
|
|
|
43c8c9 |
set_unit_handler(COMPILATION_UNIT_OBJECT,
|
|
|
43c8c9 |
amgc_cparser_no_op,true);
|
|
|
43c8c9 |
}
|
|
|
43c8c9 |
|
|
|
383aa6 |
int amgc_get_unit_ctx(
|
|
|
383aa6 |
const struct amgc_driver_ctx * dctx,
|
|
|
383aa6 |
const char * path,
|
|
|
383aa6 |
struct amgc_unit_ctx ** pctx)
|
|
|
383aa6 |
{
|
|
|
383aa6 |
struct amgc_unit_ctx_impl * ctx;
|
|
|
389acb |
FILE * ftmp;
|
|
|
389acb |
int fd;
|
|
|
383aa6 |
|
|
|
43c8c9 |
amgc_init_cparser_unit();
|
|
|
43c8c9 |
|
|
|
4671ad |
if (!dctx)
|
|
|
4671ad |
return AMGC_CUSTOM_ERROR(
|
|
|
4671ad |
dctx,AMGC_ERR_NULL_CONTEXT);
|
|
|
4671ad |
|
|
|
4671ad |
else if (!(ctx = calloc(1,sizeof(*ctx))))
|
|
|
4671ad |
return AMGC_SYSTEM_ERROR(dctx);
|
|
|
4671ad |
|
|
|
4671ad |
amgc_driver_set_ectx(
|
|
|
4671ad |
dctx,0,path);
|
|
|
383aa6 |
|
|
|
389acb |
if (strcmp(path,"-"))
|
|
|
389acb |
fd = -1;
|
|
|
4671ad |
|
|
|
389acb |
else if (!(ftmp = amgc_stdin_to_tmp(dctx)))
|
|
|
4671ad |
return amgc_free_unit_ctx_impl(
|
|
|
4671ad |
ctx,AMGC_SYSTEM_ERROR(dctx));
|
|
|
4671ad |
|
|
|
389acb |
else if ((fd = dup(fileno(ftmp))) < 0)
|
|
|
4671ad |
return amgc_free_unit_ctx_impl(
|
|
|
4671ad |
ctx,AMGC_SYSTEM_ERROR(dctx));
|
|
|
4671ad |
|
|
|
389acb |
else
|
|
|
43c8c9 |
ctx->ccunit.input = ftmp;
|
|
|
389acb |
|
|
|
bb5e47 |
if (amgc_map_input(dctx,fd,path,PROT_READ,&ctx->map))
|
|
|
4671ad |
return amgc_free_unit_ctx_impl(
|
|
|
4671ad |
ctx,AMGC_NESTED_ERROR(dctx));
|
|
|
389acb |
|
|
|
389acb |
if (fd > 0)
|
|
|
389acb |
close(fd);
|
|
|
383aa6 |
|
|
|
43c8c9 |
/* compilation unit */
|
|
|
43c8c9 |
ctx->ccunit.name = path;
|
|
|
43c8c9 |
ctx->ccunit.original_name = path;
|
|
|
43c8c9 |
ctx->ccunit.type = COMPILATION_UNIT_C;
|
|
|
fd0662 |
ctx->ccunit.standard = dctx->cctx->std;
|
|
|
43c8c9 |
|
|
|
43c8c9 |
/* parse, generate ast, generate ir */
|
|
|
43c8c9 |
if ((process_unit(ctx->cctx.ccenv,&ctx->ccunit)) == false) {
|
|
|
43c8c9 |
if (ctx->ccunit.input)
|
|
|
43c8c9 |
fclose(ctx->ccunit.input);
|
|
|
4671ad |
return amgc_free_unit_ctx_impl(
|
|
|
4671ad |
ctx,AMGC_CUSTOM_ERROR(dctx,AMGC_ERR_FLOW_ERROR));
|
|
|
43c8c9 |
}
|
|
|
43c8c9 |
|
|
|
383aa6 |
memcpy(&ctx->cctx,dctx->cctx,
|
|
|
383aa6 |
sizeof(ctx->cctx));
|
|
|
383aa6 |
|
|
|
8fca74 |
ctx->dctx = dctx;
|
|
|
f06b53 |
ctx->path = path;
|
|
|
f06b53 |
ctx->uctx.path = &ctx->path;
|
|
|
f06b53 |
ctx->uctx.map = &ctx->map;
|
|
|
f06b53 |
ctx->uctx.cctx = &ctx->cctx;
|
|
|
f06b53 |
ctx->uctx.meta = &ctx->meta;
|
|
|
f06b53 |
ctx->uctx.ccunit = &ctx->ccunit;
|
|
|
383aa6 |
|
|
|
7a622c |
if (amgc_get_unit_entities(&ctx->uctx,&ctx->meta,&ctx->entities))
|
|
|
4671ad |
return amgc_free_unit_ctx_impl(
|
|
|
4671ad |
ctx,AMGC_CUSTOM_ERROR(dctx,AMGC_ERR_FLOW_ERROR));
|
|
|
9c2028 |
|
|
|
75d04a |
ctx->uctx.entities = ctx->entities;
|
|
|
383aa6 |
*pctx = &ctx->uctx;
|
|
|
383aa6 |
return 0;
|
|
|
383aa6 |
}
|
|
|
383aa6 |
|
|
|
383aa6 |
void amgc_free_unit_ctx(struct amgc_unit_ctx * ctx)
|
|
|
383aa6 |
{
|
|
|
383aa6 |
struct amgc_unit_ctx_impl * ictx;
|
|
|
383aa6 |
uintptr_t addr;
|
|
|
383aa6 |
|
|
|
383aa6 |
if (ctx) {
|
|
|
383aa6 |
addr = (uintptr_t)ctx - offsetof(struct amgc_unit_ctx_impl,uctx);
|
|
|
383aa6 |
ictx = (struct amgc_unit_ctx_impl *)addr;
|
|
|
383aa6 |
amgc_free_unit_ctx_impl(ictx,0);
|
|
|
383aa6 |
}
|
|
|
383aa6 |
}
|