aboutsummaryrefslogtreecommitdiff
path: root/libcrystfel/src/stream.c
diff options
context:
space:
mode:
Diffstat (limited to 'libcrystfel/src/stream.c')
-rw-r--r--libcrystfel/src/stream.c83
1 files changed, 83 insertions, 0 deletions
diff --git a/libcrystfel/src/stream.c b/libcrystfel/src/stream.c
index b3ac10aa..5923af9c 100644
--- a/libcrystfel/src/stream.c
+++ b/libcrystfel/src/stream.c
@@ -80,6 +80,9 @@ struct _stream
* encountered, so stream_read_chunk() should assume a chunk is
* already in progress instead of looking for another
* marker */
+
+ long *chunk_offsets;
+ int n_chunks;
};
@@ -1148,6 +1151,8 @@ Stream *stream_open_for_read(const char *filename)
st->audit_info = NULL;
st->geometry_file = NULL;
st->in_chunk = 0;
+ st->n_chunks = 0;
+ st->chunk_offsets = NULL;
if ( strcmp(filename, "-") == 0 ) {
st->fh = stdin;
@@ -1220,6 +1225,8 @@ Stream *stream_open_fd_for_write(int fd)
st->audit_info = NULL;
st->geometry_file = NULL;
st->in_chunk = 0;
+ st->n_chunks = 0;
+ st->chunk_offsets = NULL;
st->fh = fdopen(fd, "w");
if ( st->fh == NULL ) {
@@ -1266,6 +1273,8 @@ Stream *stream_open_for_write(const char *filename)
st->audit_info = NULL;
st->geometry_file = NULL;
st->in_chunk = 0;
+ st->n_chunks = 0;
+ st->chunk_offsets = NULL;
st->fh = fopen(filename, "w");
if ( st->fh == NULL ) {
@@ -1399,3 +1408,77 @@ int stream_rewind(Stream *st)
return fseek(st->fh, 0, SEEK_SET);
}
+
+int stream_select_chunk(Stream *st, int chunk_id)
+{
+ if ( st->chunk_offsets == NULL ) return 1;
+ if ( chunk_id >= st->n_chunks ) return 1;
+ if ( fseek(st->fh, st->chunk_offsets[chunk_id], SEEK_SET) != 0 ) {
+ return 1;
+ }
+ st->in_chunk = 1;
+ return 0;
+}
+
+
+int stream_scan_chunks(Stream *st)
+{
+ long start_pos;
+ long int max_chunks = 0;
+ int done = 0;
+
+ if ( st->chunk_offsets != NULL ) {
+ ERROR("Stream has already been scanned\n");
+ return 0;
+ }
+
+ start_pos = ftell(st->fh);
+
+ /* Reset to start of stream.
+ * Also, this serves as a cursory check that the stream is
+ * actually something which can be rewound. */
+ if ( fseek(st->fh, 0, SEEK_SET) != 0 ) {
+ return 0;
+ }
+
+ do {
+
+ if ( find_start_of_chunk(st) ) {
+
+ if ( feof(st->fh) ) {
+ done = 1;
+ } else {
+ fseek(st->fh, start_pos, SEEK_SET);
+ free(st->chunk_offsets);
+ st->chunk_offsets = NULL;
+ return 0;
+ }
+ }
+
+ if ( st->n_chunks == max_chunks ) {
+
+ long *new_offsets;
+
+ max_chunks += 1024;
+ new_offsets = realloc(st->chunk_offsets,
+ max_chunks*sizeof(long));
+ if ( new_offsets == NULL ) {
+ fseek(st->fh, start_pos, SEEK_SET);
+ free(st->chunk_offsets);
+ st->chunk_offsets = NULL;
+ return 0;
+ }
+
+ st->chunk_offsets = new_offsets;
+
+ }
+
+ st->chunk_offsets[st->n_chunks++] = ftell(st->fh);
+
+ } while ( !done );
+
+ /* Reset to initial position */
+ fseek(st->fh, start_pos, SEEK_SET);
+
+ return st->n_chunks;
+}