1
0
mirror of https://github.com/vlang/v.git synced 2023-08-10 21:13:21 +03:00

os: fix rare crash in read_file. Fix #12052 (#12053)

This commit is contained in:
Larpon 2021-10-03 17:24:06 +02:00 committed by GitHub
parent 895daf297f
commit b62520af9e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -86,13 +86,19 @@ pub fn read_bytes(path string) ?[]byte {
if fsize < 0 { if fsize < 0 {
return error('ftell failed') return error('ftell failed')
} }
len := int(fsize)
// On some systems C.ftell can return values in the 64-bit range
// that, when cast to `int`, can result in values below 0.
if i64(len) < fsize {
return error('$fsize cast to int results in ${int(fsize)})')
}
C.rewind(fp) C.rewind(fp)
mut res := []byte{len: int(fsize)} mut res := []byte{len: len}
nr_read_elements := int(C.fread(res.data, fsize, 1, fp)) nr_read_elements := int(C.fread(res.data, len, 1, fp))
if nr_read_elements == 0 && fsize > 0 { if nr_read_elements == 0 && fsize > 0 {
return error('fread failed') return error('fread failed')
} }
res.trim(nr_read_elements * int(fsize)) res.trim(nr_read_elements * len)
return res return res
} }
@ -113,9 +119,15 @@ pub fn read_file(path string) ?string {
} }
// C.fseek(fp, 0, SEEK_SET) // same as `C.rewind(fp)` below // C.fseek(fp, 0, SEEK_SET) // same as `C.rewind(fp)` below
C.rewind(fp) C.rewind(fp)
allocate := int(fsize)
// On some systems C.ftell can return values in the 64-bit range
// that, when cast to `int`, can result in values below 0.
if i64(allocate) < fsize {
return error('$fsize cast to int results in ${int(fsize)})')
}
unsafe { unsafe {
mut str := malloc_noscan(int(fsize) + 1) mut str := malloc_noscan(allocate + 1)
nelements := int(C.fread(str, 1, fsize, fp)) nelements := int(C.fread(str, 1, allocate, fp))
is_eof := int(C.feof(fp)) is_eof := int(C.feof(fp))
is_error := int(C.ferror(fp)) is_error := int(C.ferror(fp))
if is_eof == 0 && is_error != 0 { if is_eof == 0 && is_error != 0 {
@ -590,7 +602,15 @@ pub fn read_file_array<T>(path string) []T {
C.rewind(fp) C.rewind(fp)
// read the actual data from the file // read the actual data from the file
len := fsize / tsize len := fsize / tsize
buf := unsafe { malloc_noscan(int(fsize)) } allocate := int(fsize)
// On some systems C.ftell can return values in the 64-bit range
// that, when cast to `int`, can result in values below 0.
if i64(allocate) < fsize {
panic('$fsize cast to int results in ${int(fsize)})')
}
buf := unsafe {
malloc_noscan(allocate)
}
nread := C.fread(buf, tsize, len, fp) nread := C.fread(buf, tsize, len, fp)
C.fclose(fp) C.fclose(fp)
return unsafe { return unsafe {