ReC98/Pipeline/zungen.c

155 lines
3.2 KiB
C

#include <stdio.h>
#include <stddef.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
#include <dir.h>
#include "./common.h"
int readparam(char *path, size_t size, FILE* fl) {
char c = '\0';
char *pathp = path;
while((--size > 0) && ((c = getc(fl)) != EOF)) {
if(('\n' == c) || (' ' == c)) {
break;
}
*(pathp++) = c;
}
if((EOF == c) && (path == pathp)) {
return 1;
}
*(pathp++) = '\0';
return 0;
}
#define MAX_COUNT 32
#define DATA_SIZE (2+MAX_COUNT*10+2+3)
static const char moveup[] = {
0xF3, 0xA4, /* REP MOVSB */
0x58, /* POP AX */
0xB8, 0x00, 0x01, /* MOV AX, 100h */
0x50, /* PUSH AX */
0xC3, /* RETN */
};
int __cdecl main(int argc, char** argv) {
static char path[MAXPATH+1];
static char names[8*MAX_COUNT];
static int entries[MAX_COUNT+1];
FILE *fl, *fo;
int count;
int size, stubsize;
int pathpos;
int i,j;
int errval;
if(argc != 4) {
printf("Usage: zungen zun_stub.bin listfile.txt outfile.bin\n");
return 1;
}
fl = fopen(argv[2],"r");
if(!fl) {
printf("Error: couldn't open listfile: %s\n", strerror(errno));
return 2;
}
fo = fopen(argv[3],"wb");
if(!fo) {
fclose(fl);
printf("Error: couldn't open outfile: %s\n", strerror(errno));
return 3;
}
/* read proc count */
if(readparam(path, sizeof(path), fl)) {
errval = 4; goto err;
}
count = atoi(path);
if(count < 0 || count > MAX_COUNT) {
printf("Error: count is too big!\n");
errval = 5; goto err;
}
/* write stub and proc count */
append(argv[1], fo);
fputw(count, fo);
/* read proc names */
memset(names, ' ', sizeof(names));
for(i=0;i<count;i++) {
if(readparam(path, sizeof(path), fl)) {
errval = 6; goto err;
}
for(j=0;j<8;j++) {
if(path[j] == 0) {
break;
}
names[j+i*8] = path[j];
}
}
/* write proc names */
if(sizeof(names) != fwrite(names, 1, sizeof(names), fo)) {
printf("Error: write error\n");
errval = 7; goto err;
}
/* save list file position for later */
pathpos = ftell(fl);
/* calc offset of first entry */
stubsize = size = filesize(argv[1]);
if(size == -1) {
errval = 8; goto err;
}
size += DATA_SIZE + 0x100;
/* build entry table */
memset(entries, 0, sizeof(entries));
for(i=0;i<count;i++) {
entries[i] = size;
if(readparam(path, sizeof(path), fl)) {
errval = 9; goto err;
}
j = filesize(path);
if(j == -1) { errval = 10; goto err; }
size += j;
}
entries[i] = size;
/* write entry table */
if(sizeof(entries) != fwrite(entries, 1, sizeof(entries), fo)) {
printf("Error: write error\n");
errval = 11; goto err;
}
/* write moveup_indirect */
fputc('\xE8', fo); /* call rel16 */
fputw(size - (stubsize + DATA_SIZE + 0x100), fo);
/* append com files */
fseek(fl, pathpos, SEEK_SET);
for(i=0;i<count;i++) {
if(readparam(path, sizeof(path), fl)) {
errval = 12; goto err;
}
if(append(path, fo)) {
errval = 13; goto err;
}
}
/* append moveup */
if(sizeof(moveup) != fwrite(moveup, 1, sizeof(moveup), fo)) {
printf("Error: write error\n");
errval = 14; goto err;
}
fclose(fl);
if(fclose(fo)) {
printf("Error: unsuccessful close\n");
return 15;
}
return 0;
err:
fclose(fl);
fclose(fo);
return errval;
}