You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
79 lines
2.0 KiB
79 lines
2.0 KiB
/* |
|
* Licensed under a two-clause BSD-style license. |
|
* See LICENSE for details. |
|
*/ |
|
|
|
#include "git-compat-util.h" |
|
#include "sliding_window.h" |
|
#include "line_buffer.h" |
|
#include "strbuf.h" |
|
|
|
static int input_error(struct line_buffer *file) |
|
{ |
|
if (!buffer_ferror(file)) |
|
return error("delta preimage ends early"); |
|
return error("cannot read delta preimage: %s", strerror(errno)); |
|
} |
|
|
|
static int skip_or_whine(struct line_buffer *file, off_t gap) |
|
{ |
|
if (buffer_skip_bytes(file, gap) != gap) |
|
return input_error(file); |
|
return 0; |
|
} |
|
|
|
static int read_to_fill_or_whine(struct line_buffer *file, |
|
struct strbuf *buf, size_t width) |
|
{ |
|
buffer_read_binary(file, buf, width - buf->len); |
|
if (buf->len != width) |
|
return input_error(file); |
|
return 0; |
|
} |
|
|
|
static int check_overflow(off_t a, size_t b) |
|
{ |
|
if (b > maximum_signed_value_of_type(off_t)) |
|
return error("unrepresentable length in delta: " |
|
"%"PRIuMAX" > OFF_MAX", (uintmax_t) b); |
|
if (signed_add_overflows(a, (off_t) b)) |
|
return error("unrepresentable offset in delta: " |
|
"%"PRIuMAX" + %"PRIuMAX" > OFF_MAX", |
|
(uintmax_t) a, (uintmax_t) b); |
|
return 0; |
|
} |
|
|
|
int move_window(struct sliding_view *view, off_t off, size_t width) |
|
{ |
|
off_t file_offset; |
|
assert(view); |
|
assert(view->width <= view->buf.len); |
|
assert(!check_overflow(view->off, view->buf.len)); |
|
|
|
if (check_overflow(off, width)) |
|
return -1; |
|
if (off < view->off || off + width < view->off + view->width) |
|
return error("invalid delta: window slides left"); |
|
if (view->max_off >= 0 && view->max_off < off + width) |
|
return error("delta preimage ends early"); |
|
|
|
file_offset = view->off + view->buf.len; |
|
if (off < file_offset) { |
|
/* Move the overlapping region into place. */ |
|
strbuf_remove(&view->buf, 0, off - view->off); |
|
} else { |
|
/* Seek ahead to skip the gap. */ |
|
if (skip_or_whine(view->file, off - file_offset)) |
|
return -1; |
|
strbuf_setlen(&view->buf, 0); |
|
} |
|
|
|
if (view->buf.len > width) |
|
; /* Already read. */ |
|
else if (read_to_fill_or_whine(view->file, &view->buf, width)) |
|
return -1; |
|
|
|
view->off = off; |
|
view->width = width; |
|
return 0; |
|
}
|
|
|