2022-11-06 10:22:27 +00:00
|
|
|
#define _IN_MINITAR
|
2022-11-05 19:10:48 +00:00
|
|
|
#include "tar.h"
|
2022-11-06 10:02:26 +00:00
|
|
|
#include "minitar.h"
|
|
|
|
#include <stdlib.h>
|
2022-11-06 10:51:03 +00:00
|
|
|
#include <string.h>
|
2022-11-05 19:10:48 +00:00
|
|
|
|
2022-11-06 13:23:35 +00:00
|
|
|
// all of these are defined in util.c
|
2022-11-06 13:21:32 +00:00
|
|
|
int minitar_read_header(struct minitar*, struct tar_header*);
|
|
|
|
int minitar_validate_header(struct tar_header*);
|
|
|
|
void minitar_parse_tar_header(struct tar_header*, struct minitar_entry_metadata*);
|
|
|
|
struct minitar_entry* minitar_dup_entry(struct minitar_entry*);
|
2022-11-06 13:23:35 +00:00
|
|
|
char* minitar_read_file_contents(struct minitar_entry_metadata*, struct minitar*);
|
2022-11-05 17:52:51 +00:00
|
|
|
|
2022-11-06 13:21:32 +00:00
|
|
|
struct minitar* minitar_open(const char* pathname)
|
2022-11-05 17:52:51 +00:00
|
|
|
{
|
2022-11-06 13:21:32 +00:00
|
|
|
FILE* fp = fopen(pathname, "rb"); // On some systems, this might be necessary to read the file properly.
|
2022-11-06 10:02:26 +00:00
|
|
|
if (!fp) return NULL;
|
2022-11-05 17:52:51 +00:00
|
|
|
struct minitar* mp = malloc(sizeof(struct minitar));
|
2022-11-06 10:02:26 +00:00
|
|
|
if (!mp)
|
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
return NULL;
|
|
|
|
}
|
2022-11-05 17:52:51 +00:00
|
|
|
mp->stream = fp;
|
|
|
|
return mp;
|
|
|
|
}
|
|
|
|
|
|
|
|
int minitar_close(struct minitar* mp)
|
|
|
|
{
|
|
|
|
int rc = fclose(mp->stream);
|
|
|
|
free(mp);
|
2022-11-06 10:02:26 +00:00
|
|
|
if (rc) return rc;
|
2022-11-05 17:52:51 +00:00
|
|
|
return 0;
|
2022-11-05 19:10:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct minitar_entry* minitar_attempt_read_entry(struct minitar* mp, int* valid)
|
|
|
|
{
|
|
|
|
struct minitar_entry entry;
|
|
|
|
struct tar_header hdr;
|
2022-11-06 10:02:26 +00:00
|
|
|
if (!minitar_read_header(mp, &hdr))
|
2022-11-05 19:10:48 +00:00
|
|
|
{
|
|
|
|
*valid = 1; // we are at end-of-file
|
|
|
|
return NULL;
|
|
|
|
}
|
2022-11-06 10:02:26 +00:00
|
|
|
if (!minitar_validate_header(&hdr))
|
2022-11-05 19:10:48 +00:00
|
|
|
{
|
|
|
|
*valid = 0;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
*valid = 1;
|
|
|
|
minitar_parse_tar_header(&hdr, &entry.metadata);
|
2022-11-06 13:23:35 +00:00
|
|
|
char* buf = minitar_read_file_contents(&entry.metadata, mp);
|
2022-11-06 10:02:26 +00:00
|
|
|
if (!buf) return NULL;
|
2022-11-06 09:33:37 +00:00
|
|
|
entry.ptr = buf;
|
2022-11-05 19:10:48 +00:00
|
|
|
return minitar_dup_entry(&entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct minitar_entry* minitar_read_entry(struct minitar* mp)
|
|
|
|
{
|
|
|
|
int valid;
|
|
|
|
struct minitar_entry* result;
|
|
|
|
do {
|
|
|
|
result = minitar_attempt_read_entry(mp, &valid);
|
2022-11-06 10:02:26 +00:00
|
|
|
} while (!valid);
|
2022-11-05 19:10:48 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-11-06 10:22:27 +00:00
|
|
|
void minitar_rewind(struct minitar* mp)
|
|
|
|
{
|
|
|
|
rewind(mp->stream);
|
|
|
|
}
|
|
|
|
|
2022-11-05 19:10:48 +00:00
|
|
|
void minitar_free_entry(struct minitar_entry* entry)
|
|
|
|
{
|
2022-11-06 09:33:37 +00:00
|
|
|
free(entry->ptr);
|
|
|
|
free(entry);
|
2022-11-06 10:51:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct minitar_entry* minitar_find_by_name(struct minitar* mp, const char* name)
|
|
|
|
{
|
|
|
|
struct minitar_entry* entry;
|
|
|
|
do {
|
|
|
|
entry = minitar_read_entry(mp);
|
|
|
|
if (entry)
|
|
|
|
{
|
|
|
|
if (!strcmp(entry->metadata.name, name)) return entry;
|
|
|
|
minitar_free_entry(entry);
|
|
|
|
}
|
|
|
|
} while (entry);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct minitar_entry* minitar_find_any_of(struct minitar* mp, enum minitar_file_type type)
|
|
|
|
{
|
|
|
|
struct minitar_entry* entry;
|
|
|
|
do {
|
|
|
|
entry = minitar_read_entry(mp);
|
|
|
|
if (entry)
|
|
|
|
{
|
|
|
|
if (entry->metadata.type == type) return entry;
|
|
|
|
minitar_free_entry(entry);
|
|
|
|
}
|
|
|
|
} while (entry);
|
|
|
|
return NULL;
|
2022-11-05 17:52:51 +00:00
|
|
|
}
|