aboutsummaryrefslogtreecommitdiff
path: root/src/libutil
diff options
context:
space:
mode:
authorEelco Dolstra <e.dolstra@tudelft.nl>2009-03-22 17:36:43 +0000
committerEelco Dolstra <e.dolstra@tudelft.nl>2009-03-22 17:36:43 +0000
commit77d272623fb4fd57cf27d4b92a7dc1713a2d4098 (patch)
tree37c40b30768dfe348527a93237652e041280419a /src/libutil
parent7e05b8b75e0f4b370cc7d4b78b3fb18a3678b360 (diff)
* NAR archives: handle files larger than 2^32 bytes. Previously it
would just silently store only (fileSize % 2^32) bytes. * Use posix_fallocate if available when unpacking archives. * Provide a better error message when trying to unpack something that isn't a NAR archive.
Diffstat (limited to 'src/libutil')
-rw-r--r--src/libutil/archive.cc45
-rw-r--r--src/libutil/archive.hh1
-rw-r--r--src/libutil/serialise.cc4
-rw-r--r--src/libutil/serialise.hh3
4 files changed, 40 insertions, 13 deletions
diff --git a/src/libutil/archive.cc b/src/libutil/archive.cc
index 30b94951b..3142aa929 100644
--- a/src/libutil/archive.cc
+++ b/src/libutil/archive.cc
@@ -2,6 +2,7 @@
#include <algorithm>
#include <vector>
+#define _XOPEN_SOURCE 600
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
@@ -11,6 +12,8 @@
#include "archive.hh"
#include "util.hh"
+#include "config.h"
+
namespace nix {
@@ -47,17 +50,17 @@ static void dumpEntries(const Path & path, Sink & sink, PathFilter & filter)
}
-static void dumpContents(const Path & path, unsigned int size,
+static void dumpContents(const Path & path, off_t size,
Sink & sink)
{
writeString("contents", sink);
- writeInt(size, sink);
+ writeLongLong(size, sink);
AutoCloseFD fd = open(path.c_str(), O_RDONLY);
if (fd == -1) throw SysError(format("opening file `%1%'") % path);
unsigned char buf[65536];
- unsigned int left = size;
+ off_t left = size;
while (left > 0) {
size_t n = left > sizeof(buf) ? sizeof(buf) : left;
@@ -101,7 +104,7 @@ static void dump(const Path & path, Sink & sink, PathFilter & filter)
writeString(readLink(path), sink);
}
- else throw Error("unknown file type: " + path);
+ else throw Error(format("file `%1%' has an unknown type") % path);
writeString(")", sink);
}
@@ -114,9 +117,9 @@ void dumpPath(const Path & path, Sink & sink, PathFilter & filter)
}
-static Error badArchive(string s)
+static SerialisationError badArchive(string s)
{
- return Error("bad archive: " + s);
+ return SerialisationError("bad archive: " + s);
}
@@ -162,14 +165,17 @@ static void parseEntry(ParseSink & sink, Source & source, const Path & path)
static void parseContents(ParseSink & sink, Source & source, const Path & path)
{
- unsigned int size = readInt(source);
- unsigned int left = size;
+ unsigned long long size = readLongLong(source);
+
+ sink.preallocateContents(size);
+
+ unsigned long long left = size;
unsigned char buf[65536];
while (left) {
checkInterrupt();
unsigned int n = sizeof(buf);
- if (n > left) n = left;
+ if ((unsigned long long) n > left) n = left;
source(buf, n);
sink.receiveContents(buf, n);
left -= n;
@@ -248,8 +254,15 @@ static void parse(ParseSink & sink, Source & source, const Path & path)
void parseDump(ParseSink & sink, Source & source)
{
- if (readString(source) != archiveVersion1)
- throw badArchive("expected Nix archive");
+ string version;
+ try {
+ version = readString(source);
+ } catch (SerialisationError & e) {
+ /* This generally means the integer at the start couldn't be
+ decoded. Ignore and throw the exception below. */
+ }
+ if (version != archiveVersion1)
+ throw badArchive("input doesn't look like a Nix archive");
parse(sink, source, "");
}
@@ -282,6 +295,16 @@ struct RestoreSink : ParseSink
throw SysError("fchmod");
}
+ void preallocateContents(unsigned long long len)
+ {
+#if HAVE_POSIX_FALLOCATE
+ if (len) {
+ errno = posix_fallocate(fd, 0, len);
+ if (errno) throw SysError(format("preallocating file of %1% bytes") % len);
+ }
+#endif
+ }
+
void receiveContents(unsigned char * data, unsigned int len)
{
writeFull(fd, data, len);
diff --git a/src/libutil/archive.hh b/src/libutil/archive.hh
index 06f09cd1e..fff620313 100644
--- a/src/libutil/archive.hh
+++ b/src/libutil/archive.hh
@@ -62,6 +62,7 @@ struct ParseSink
virtual void createRegularFile(const Path & path) { };
virtual void isExecutable() { };
+ virtual void preallocateContents(unsigned long long size) { };
virtual void receiveContents(unsigned char * data, unsigned int len) { };
virtual void createSymlink(const Path & path, const string & target) { };
diff --git a/src/libutil/serialise.cc b/src/libutil/serialise.cc
index c13e8c7e3..9b4222713 100644
--- a/src/libutil/serialise.cc
+++ b/src/libutil/serialise.cc
@@ -80,7 +80,7 @@ void readPadding(unsigned int len, Source & source)
unsigned int n = 8 - (len % 8);
source(zero, n);
for (unsigned int i = 0; i < n; i++)
- if (zero[i]) throw Error("non-zero padding");
+ if (zero[i]) throw SerialisationError("non-zero padding");
}
}
@@ -90,7 +90,7 @@ unsigned int readInt(Source & source)
unsigned char buf[8];
source(buf, sizeof(buf));
if (buf[4] || buf[5] || buf[6] || buf[7])
- throw Error("implementation cannot deal with > 32-bit integers");
+ throw SerialisationError("implementation cannot deal with > 32-bit integers");
return
buf[0] |
(buf[1] << 8) |
diff --git a/src/libutil/serialise.hh b/src/libutil/serialise.hh
index 6a74c4cf6..0e797d63b 100644
--- a/src/libutil/serialise.hh
+++ b/src/libutil/serialise.hh
@@ -106,6 +106,9 @@ string readString(Source & source);
StringSet readStringSet(Source & source);
+MakeError(SerialisationError, Error)
+
+
}