toml: isolate, fix and regress-test sumtype cast causing memory corruption (#12329)

pull/12334/head
Larpon 2021-10-28 18:57:30 +02:00 committed by GitHub
parent fa02418a55
commit 5e4594a121
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 31 additions and 14 deletions

View File

@ -21,7 +21,11 @@ pub type Any = Null
// string returns `Any` as a string. // string returns `Any` as a string.
pub fn (a Any) string() string { pub fn (a Any) string() string {
match a { match a {
string { return a as string } // NOTE if `.clone()` is not used here:
// string { return a as string }
// ... certain call-patterns to this function will cause a memory corruption.
// See `tests/toml_memory_corruption_test.v` for a matching regression test.
string { return (a as string).clone() }
time.Time { return a.format_ss_micro() } time.Time { return a.format_ss_micro() }
else { return a.str() } else { return a.str() }
} }

View File

@ -35,18 +35,13 @@ fn test_parse_compact_text() {
assert title == toml.Any('TOML Example') assert title == toml.Any('TOML Example')
assert title as string == 'TOML Example' assert title as string == 'TOML Example'
owner := toml_doc.value('owner') as map[string]toml.Any
any_name := owner.value('name') or { panic(err) }
assert any_name.string() == 'Tom Preston-Werner'
database := toml_doc.value('database') as map[string]toml.Any database := toml_doc.value('database') as map[string]toml.Any
db_serv := database['server'] or { db_serv := database['server'] or {
panic('could not access "server" index in "database" variable') panic('could not access "server" index in "database" variable')
} }
assert db_serv as string == '192.168.1.1' assert db_serv as string == '192.168.1.1'
// TODO BUG depending on WHAT directory the tests is run from, this one assert sometimes fail?!?! assert toml_doc.value('owner.name') as string == 'Tom Preston-Werner'
// assert toml_doc.value('owner.name') as string == 'Tom Preston-Werner'
assert toml_doc.value('database.server') as string == '192.168.1.1' assert toml_doc.value('database.server') as string == '192.168.1.1'

View File

@ -0,0 +1,24 @@
// This tests the `toml` module for a known memory corruption.
// The BUG shows below if no string `.clone()` nor any garbage-collection is done...
import os
import toml
const toml_text = os.read_file(os.real_path(os.join_path(os.dir(@FILE), 'testdata', 'toml_test')) +
'.toml') or { panic(err) }
fn test_toml_known_memory_corruption() {
toml_doc := toml.parse(toml_text) or { panic(err) }
owner := toml_doc.value('owner') as map[string]toml.Any
any_name := owner.value('name') or { panic(err) }
// This assert code path will cause the corruption.
assert any_name.string() == 'Tom Preston-Werner'
// This code then triggered the bug before the fix.
// Also see note in toml/any.v in function `pub fn (a Any) string() string`
assert toml_doc.value('owner.name') as string == 'Tom Preston-Werner'
// Repeat the pattern
assert any_name.string() == 'Tom Preston-Werner'
assert toml_doc.value('owner.name') as string == 'Tom Preston-Werner'
}

View File

@ -24,18 +24,13 @@ fn test_toml() {
assert title == toml.Any('TOML Example') assert title == toml.Any('TOML Example')
assert title as string == 'TOML Example' assert title as string == 'TOML Example'
owner := toml_doc.value('owner') as map[string]toml.Any
any_name := owner.value('name') or { panic(err) }
assert any_name.string() == 'Tom Preston-Werner'
database := toml_doc.value('database') as map[string]toml.Any database := toml_doc.value('database') as map[string]toml.Any
db_serv := database['server'] or { db_serv := database['server'] or {
panic('could not access "server" index in "database" variable') panic('could not access "server" index in "database" variable')
} }
assert db_serv as string == '192.168.1.1' assert db_serv as string == '192.168.1.1'
// TODO BUG depending on WHAT directory the tests is run from, this one assert sometimes fail?!?! assert toml_doc.value('owner.name') as string == 'Tom Preston-Werner'
// assert toml_doc.value('owner.name') as string == 'Tom Preston-Werner'
assert toml_doc.value('database.server') as string == '192.168.1.1' assert toml_doc.value('database.server') as string == '192.168.1.1'

View File

@ -92,7 +92,6 @@ pub fn (d Doc) to_json() string {
// value queries a value from the TOML document. // value queries a value from the TOML document.
pub fn (d Doc) value(key string) Any { pub fn (d Doc) value(key string) Any {
values := d.ast.table as map[string]ast.Value values := d.ast.table as map[string]ast.Value
// any_values := d.ast_to_any(values) as map[string]Any
return d.get_map_value_as_any(values, key) return d.get_map_value_as_any(values, key)
} }