summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/tpax/tpax.h5
-rw-r--r--project/common.mk1
-rw-r--r--src/logic/tpax_file_create_memory_snapshot.c101
-rw-r--r--src/output/tpax_output_error.c2
4 files changed, 109 insertions, 0 deletions
diff --git a/include/tpax/tpax.h b/include/tpax/tpax.h
index c39f721..8d6183b 100644
--- a/include/tpax/tpax.h
+++ b/include/tpax/tpax.h
@@ -66,6 +66,8 @@ enum tpax_custom_error {
TPAX_ERR_NULL_CONTEXT,
TPAX_ERR_BAD_DATA,
TPAX_ERR_FORBIDDEN_PATH,
+ TPAX_ERR_FILE_CHANGED,
+ TPAX_ERR_REGION_SIZE,
TPAX_ERR_CAP,
};
@@ -161,6 +163,9 @@ tpax_api int tpax_output_error_vector (const struct tpax_driver_ctx *);
tpax_api int tpax_init_ustar_header (const struct tpax_driver_ctx *, const char *, const struct stat *,
const char *, struct tpax_ustar_header *);
+tpax_api int tpax_file_create_memory_snapshot (const struct tpax_driver_ctx *, const char *,
+ const struct stat *, void *);
+
/* package info */
tpax_api const struct tpax_source_version * tpax_source_version(void);
diff --git a/project/common.mk b/project/common.mk
index 14d6ec8..da9fa26 100644
--- a/project/common.mk
+++ b/project/common.mk
@@ -5,6 +5,7 @@ API_SRCS = \
src/helper/tpax_path_copy.c \
src/helper/tpax_stat_compare.c \
src/logic/tpax_init_ustar_header.c \
+ src/logic/tpax_file_create_memory_snapshot.c \
src/output/tpax_output_error.c \
src/skin/tpax_skin_default.c \
diff --git a/src/logic/tpax_file_create_memory_snapshot.c b/src/logic/tpax_file_create_memory_snapshot.c
new file mode 100644
index 0000000..7db21ea
--- /dev/null
+++ b/src/logic/tpax_file_create_memory_snapshot.c
@@ -0,0 +1,101 @@
+/******************************************************/
+/* tpax: a topological pax implementation */
+/* Copyright (C) 2020 Z. Gilboa */
+/* Released under GPLv2 and GPLv3; see COPYING.TPAX. */
+/******************************************************/
+
+#include <stdint.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+#include <fcntl.h>
+#include <errno.h>
+#include <grp.h>
+#include <pwd.h>
+#include <sys/stat.h>
+
+#include <tpax/tpax.h>
+#include <tpax/tpax_specs.h>
+#include "tpax_driver_impl.h"
+#include "tpax_errinfo_impl.h"
+
+#ifndef ssizeof
+#define ssizeof(x) (ssize_t)(sizeof(x))
+#endif
+
+int tpax_file_create_memory_snapshot(
+ const struct tpax_driver_ctx * dctx,
+ const char * path,
+ const struct stat * srcst,
+ void * addr)
+{
+ int fd;
+ char * ch;
+ char * cap;
+ ssize_t nread;
+ struct stat dstst;
+
+ /* record errors */
+ tpax_driver_set_ectx(
+ dctx,0,path);
+
+ /* memory snapshot internal limit */
+ if (srcst->st_size >= 0x80000000)
+ return TPAX_CUSTOM_ERROR(dctx,TPAX_ERR_REGION_SIZE);
+
+ /* open */
+ fd = openat(
+ tpax_driver_fdcwd(dctx),path,
+ O_CLOEXEC|O_NOCTTY|O_NOFOLLOW);
+
+ if (fd < 0)
+ return TPAX_SYSTEM_ERROR(dctx);
+
+ /* stat compare */
+ if ((fstat(fd,&dstst)) < 0) {
+ close(fd);
+ return TPAX_SYSTEM_ERROR(dctx);
+
+ } else if (tpax_stat_compare(srcst,&dstst)) {
+ close(fd);
+ return TPAX_CUSTOM_ERROR(dctx,TPAX_ERR_FILE_CHANGED);
+ }
+
+ /* read loop */
+ ch = addr;
+ cap = &ch[srcst->st_size];
+
+ while (ch < cap) {
+ nread = read(fd,ch,cap-ch);
+
+ while ((nread < 0) && (errno == EINTR))
+ nread = read(fd,ch,cap-ch);
+
+ if (nread < 0) {
+ close(fd);
+ return TPAX_SYSTEM_ERROR(dctx);
+
+ } else if (nread == 0) {
+ close(fd);
+ return TPAX_CUSTOM_ERROR(dctx,TPAX_ERR_FLOW_ERROR);
+
+ } else {
+ ch += nread;
+ }
+ }
+
+ /* stat compare */
+ if ((fstat(fd,&dstst)) < 0) {
+ close(fd);
+ return TPAX_SYSTEM_ERROR(dctx);
+
+ } else if (tpax_stat_compare(srcst,&dstst)) {
+ close(fd);
+ return TPAX_CUSTOM_ERROR(dctx,TPAX_ERR_FILE_CHANGED);
+ }
+
+ /* yay */
+ close(fd);
+
+ return 0;
+}
diff --git a/src/output/tpax_output_error.c b/src/output/tpax_output_error.c
index 69c4938..de09f21 100644
--- a/src/output/tpax_output_error.c
+++ b/src/output/tpax_output_error.c
@@ -26,6 +26,8 @@ static const char * const tpax_error_strings[TPAX_ERR_CAP] = {
[TPAX_ERR_FLEE_ERROR] = "flees and bugs and cats and mice",
[TPAX_ERR_NULL_CONTEXT] = "null driver or unit context",
[TPAX_ERR_BAD_DATA] = "corrupt or wrong daata",
+ [TPAX_ERR_FILE_CHANGED] = "file has changed",
+ [TPAX_ERR_REGION_SIZE] = "file too large for a memory snapshot",
};
static const char * tpax_output_error_header(const struct tpax_error_info * erri)