1
0
mirror of git://projects.qi-hardware.com/eda-tools.git synced 2024-11-23 01:32:49 +02:00
eda-tools/eeshow/file.c

163 lines
3.2 KiB
C

/*
* file.c - Open and read a file
*
* Written 2016 by Werner Almesberger
* Copyright 2016 by Werner Almesberger
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*/
#include <stdbool.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "util.h"
#include "main.h"
#include "git-file.h"
#include "file.h"
bool file_cat(const struct file *file, void *user, const char *line)
{
printf("%s\n", line);
return 1;
}
static bool try_related(struct file *file)
{
const char *related;
const char *slash;
char *tmp;
unsigned len;
if (!file->related)
return 0;
if (*file->name == '/')
return 0;
related = file->related->name;
slash = strrchr(related, '/');
if (!slash)
return 0;
len = slash + 1 - related;
tmp = alloc_size(len + strlen(file->name) + 1);
memcpy(tmp, related, len);
strcpy(tmp + len, file->name);
file->file = fopen(tmp, "r");
if (!file->file) {
free(tmp);
return 0;
}
if (verbose)
fprintf(stderr, "reading %s\n", tmp);
free((char *) file->name);
file->name = tmp;
return 1;
}
/*
* @@@ logic isn't quite complete yet. It should go something like this:
*
* - if there is no related item,
* - try file,
* - if there is a colon, try VCS,
* - give up
* - if there is a related item,
* - if it is a VCS,
* - try the revision matching or predating (if different repo) that of the
* related repo,
( - fall through and try as if it was a file
* - try opening as file. If this fails,
* - if the name of the file to open is absolute, give up
* - try `dirname related`/file_ope_open
* - give up
*
* @@@ should we see if non-VCS file is in a VCS ? E.g.,
* /home/my-libs/foo.lib 1234:/home/my-project/foo.sch
* or maybe use : as indictor for VCS, i.e.,
* :/home/my-libs/foo.lib ...
*
* @@@ explicit revision should always win over related.
*/
void file_open(struct file *file, const char *name, const struct file *related)
{
char *colon, *tmp;
file->name = stralloc(name);
file->lineno = 0;
file->related = related;
file->vcs = NULL;
file->file = fopen(name, "r");
if (file->file) {
if (verbose)
fprintf(stderr, "reading %s\n", name);
return;
}
if (try_related(file))
return;
if (verbose)
perror(name);
colon = strchr(name, ':');
if (!colon) {
if (!verbose)
perror(name);
exit(1);
}
tmp = stralloc(name);
tmp[colon - name] = 0;
file->vcs = vcs_git_open(tmp, colon + 1);
if (!file->vcs) {
fprintf(stderr, "could not open %s:%s\n", tmp, colon + 1);
exit(1);
}
free(tmp);
}
void file_read(struct file *file,
bool (*parse)(const struct file *file, void *user, const char *line),
void *user)
{
char buf[1000];
char *nl;
if (file->vcs) {
vcs_read(file->vcs, file, parse, user);
return;
}
while (fgets(buf, sizeof(buf), file->file)) {
nl = strchr(buf, '\n');
if (nl)
*nl = 0;
file->lineno++;
if (!parse(file, user, buf))
break;
}
}
void file_close(struct file *file)
{
if (file->file)
fclose(file->file);
if (file->vcs)
vcs_close(file->vcs);
free((char *) file->name);
}