cgen: cleanup `Gen.gen_*_equality_fn` (#10633)

pull/10636/head
Enzo 2021-07-01 14:42:49 +02:00 committed by GitHub
parent aa8d9b6cbc
commit d0a09579cc
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 192 additions and 190 deletions

View File

@ -5,200 +5,202 @@ module c
import strings import strings
import v.ast import v.ast
fn (mut g Gen) gen_sumtype_equality_fn(left ast.Type) string { fn (mut g Gen) gen_sumtype_equality_fn(left_type ast.Type) string {
ptr_typ := g.typ(left).trim('*') left := g.unwrap(left_type)
if ptr_typ in g.sumtype_fn_definitions { ptr_styp := g.typ(left.typ.set_nr_muls(0))
return ptr_typ if ptr_styp in g.sumtype_fn_definitions {
return ptr_styp
} }
g.sumtype_fn_definitions << ptr_typ g.sumtype_fn_definitions << ptr_styp
left_sym := g.table.get_type_symbol(left) info := left.sym.sumtype_info()
info := left_sym.sumtype_info() g.type_definitions.writeln('static bool ${ptr_styp}_sumtype_eq($ptr_styp a, $ptr_styp b); // auto')
g.type_definitions.writeln('static bool ${ptr_typ}_sumtype_eq($ptr_typ a, $ptr_typ b); // auto')
mut fn_builder := strings.new_builder(512)
fn_builder.writeln('static bool ${ptr_typ}_sumtype_eq($ptr_typ a, $ptr_typ b) {')
mut fn_builder := strings.new_builder(512)
fn_builder.writeln('static bool ${ptr_styp}_sumtype_eq($ptr_styp a, $ptr_styp b) {')
fn_builder.writeln('\tif (a._typ != b._typ) { return false; }') fn_builder.writeln('\tif (a._typ != b._typ) { return false; }')
for typ in info.variants { for typ in info.variants {
sym := g.table.get_type_symbol(typ) variant := g.unwrap(typ)
fn_builder.writeln('\tif (a._typ == $typ) {') fn_builder.writeln('\tif (a._typ == $typ) {')
name := '_$sym.cname' name := '_$variant.sym.cname'
if sym.kind == .string { if variant.sym.kind == .string {
fn_builder.writeln('\t\tif (!string__eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn string__eq(*a.$name, *b.$name);')
} else if sym.kind == .sum_type && !typ.is_ptr() { } else if variant.sym.kind == .sum_type && !typ.is_ptr() {
eq_fn := g.gen_sumtype_equality_fn(typ) eq_fn := g.gen_sumtype_equality_fn(typ)
fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn ${eq_fn}_sumtype_eq(*a.$name, *b.$name);')
} else if sym.kind == .struct_ && !typ.is_ptr() { } else if variant.sym.kind == .struct_ && !typ.is_ptr() {
eq_fn := g.gen_struct_equality_fn(typ) eq_fn := g.gen_struct_equality_fn(typ)
fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn ${eq_fn}_struct_eq(*a.$name, *b.$name);')
} else if sym.kind == .array && !typ.is_ptr() { } else if variant.sym.kind == .array && !typ.is_ptr() {
eq_fn := g.gen_array_equality_fn(typ) eq_fn := g.gen_array_equality_fn(typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn ${eq_fn}_arr_eq(*a.$name, *b.$name);')
} else if sym.kind == .array_fixed && !typ.is_ptr() { } else if variant.sym.kind == .array_fixed && !typ.is_ptr() {
eq_fn := g.gen_fixed_array_equality_fn(typ) eq_fn := g.gen_fixed_array_equality_fn(typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn ${eq_fn}_arr_eq(*a.$name, *b.$name);')
} else if sym.kind == .map && !typ.is_ptr() { } else if variant.sym.kind == .map && !typ.is_ptr() {
eq_fn := g.gen_map_equality_fn(typ) eq_fn := g.gen_map_equality_fn(typ)
fn_builder.writeln('\t\tif (!${eq_fn}_map_eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn ${eq_fn}_map_eq(*a.$name, *b.$name);')
} else if sym.kind == .alias && !typ.is_ptr() { } else if variant.sym.kind == .alias && !typ.is_ptr() {
eq_fn := g.gen_alias_equality_fn(typ) eq_fn := g.gen_alias_equality_fn(typ)
fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq(*a.$name, *b.$name)) {') fn_builder.writeln('\t\treturn ${eq_fn}_alias_eq(*a.$name, *b.$name);')
} else if sym.kind == .function { } else if variant.sym.kind == .function {
fn_builder.writeln('\t\tif (*((voidptr*)(*a.$name)) != *((voidptr*)(*b.$name))) {') fn_builder.writeln('\t\treturn *((voidptr*)(*a.$name)) == *((voidptr*)(*b.$name));')
} else { } else {
fn_builder.writeln('\t\tif (*a.$name != *b.$name) {') fn_builder.writeln('\t\treturn *a.$name == *b.$name;')
} }
fn_builder.writeln('\t\t\treturn false;')
fn_builder.writeln('\t\t}')
fn_builder.writeln('\t\treturn true;')
fn_builder.writeln('\t}') fn_builder.writeln('\t}')
} }
fn_builder.writeln('\treturn false;') fn_builder.writeln('\treturn false;')
fn_builder.writeln('}') fn_builder.writeln('}')
g.auto_fn_definitions << fn_builder.str() g.auto_fn_definitions << fn_builder.str()
return ptr_typ return ptr_styp
} }
fn (mut g Gen) gen_struct_equality_fn(left ast.Type) string { fn (mut g Gen) gen_struct_equality_fn(left_type ast.Type) string {
ptr_typ := g.typ(left).trim('*') left := g.unwrap(left_type)
if ptr_typ in g.struct_fn_definitions { ptr_styp := g.typ(left.typ.set_nr_muls(0))
return ptr_typ if ptr_styp in g.struct_fn_definitions {
return ptr_styp
} }
g.struct_fn_definitions << ptr_typ g.struct_fn_definitions << ptr_styp
left_sym := g.table.get_type_symbol(left) info := left.sym.struct_info()
info := left_sym.struct_info() g.type_definitions.writeln('static bool ${ptr_styp}_struct_eq($ptr_styp a, $ptr_styp b); // auto')
g.type_definitions.writeln('static bool ${ptr_typ}_struct_eq($ptr_typ a, $ptr_typ b); // auto')
mut fn_builder := strings.new_builder(512) mut fn_builder := strings.new_builder(512)
defer { defer {
g.auto_fn_definitions << fn_builder.str() g.auto_fn_definitions << fn_builder.str()
} }
fn_builder.writeln('static bool ${ptr_typ}_struct_eq($ptr_typ a, $ptr_typ b) {') fn_builder.writeln('static bool ${ptr_styp}_struct_eq($ptr_styp a, $ptr_styp b) {')
// overloaded // overloaded
if left_sym.has_method('==') { if left.sym.has_method('==') {
fn_builder.writeln('\treturn ${ptr_typ}__eq(a, b);') fn_builder.writeln('\treturn ${ptr_styp}__eq(a, b);')
fn_builder.writeln('}') fn_builder.writeln('}')
return ptr_typ return ptr_styp
} }
for field in info.fields { fn_builder.write_string('\treturn ')
sym := g.table.get_type_symbol(field.typ) if info.fields.len > 0 {
if sym.kind == .string { for i, field in info.fields {
fn_builder.writeln('\tif (!string__eq(a.$field.name, b.$field.name)) {') if i > 0 {
} else if sym.kind == .sum_type && !field.typ.is_ptr() { fn_builder.write_string('\n\t\t&& ')
}
field_type := g.unwrap(field.typ)
if field_type.sym.kind == .string {
fn_builder.write_string('string__eq(a.$field.name, b.$field.name)')
} else if field_type.sym.kind == .sum_type && !field.typ.is_ptr() {
eq_fn := g.gen_sumtype_equality_fn(field.typ) eq_fn := g.gen_sumtype_equality_fn(field.typ)
fn_builder.writeln('\tif (!${eq_fn}_sumtype_eq(a.$field.name, b.$field.name)) {') fn_builder.write_string('${eq_fn}_sumtype_eq(a.$field.name, b.$field.name)')
} else if sym.kind == .struct_ && !field.typ.is_ptr() { } else if field_type.sym.kind == .struct_ && !field.typ.is_ptr() {
eq_fn := g.gen_struct_equality_fn(field.typ) eq_fn := g.gen_struct_equality_fn(field.typ)
fn_builder.writeln('\tif (!${eq_fn}_struct_eq(a.$field.name, b.$field.name)) {') fn_builder.write_string('${eq_fn}_struct_eq(a.$field.name, b.$field.name)')
} else if sym.kind == .array && !field.typ.is_ptr() { } else if field_type.sym.kind == .array && !field.typ.is_ptr() {
eq_fn := g.gen_array_equality_fn(field.typ) eq_fn := g.gen_array_equality_fn(field.typ)
fn_builder.writeln('\tif (!${eq_fn}_arr_eq(a.$field.name, b.$field.name)) {') fn_builder.write_string('${eq_fn}_arr_eq(a.$field.name, b.$field.name)')
} else if sym.kind == .array_fixed && !field.typ.is_ptr() { } else if field_type.sym.kind == .array_fixed && !field.typ.is_ptr() {
eq_fn := g.gen_fixed_array_equality_fn(field.typ) eq_fn := g.gen_fixed_array_equality_fn(field.typ)
fn_builder.writeln('\tif (!${eq_fn}_arr_eq(a.$field.name, b.$field.name)) {') fn_builder.write_string('${eq_fn}_arr_eq(a.$field.name, b.$field.name)')
} else if sym.kind == .map && !field.typ.is_ptr() { } else if field_type.sym.kind == .map && !field.typ.is_ptr() {
eq_fn := g.gen_map_equality_fn(field.typ) eq_fn := g.gen_map_equality_fn(field.typ)
fn_builder.writeln('\tif (!${eq_fn}_map_eq(a.$field.name, b.$field.name)) {') fn_builder.write_string('${eq_fn}_map_eq(a.$field.name, b.$field.name)')
} else if sym.kind == .alias && !field.typ.is_ptr() { } else if field_type.sym.kind == .alias && !field.typ.is_ptr() {
eq_fn := g.gen_alias_equality_fn(field.typ) eq_fn := g.gen_alias_equality_fn(field.typ)
fn_builder.writeln('\tif (!${eq_fn}_alias_eq(a.$field.name, b.$field.name)) {') fn_builder.write_string('${eq_fn}_alias_eq(a.$field.name, b.$field.name)')
} else if sym.kind == .function { } else if field_type.sym.kind == .function {
fn_builder.writeln('\tif (*((voidptr*)(a.$field.name)) != *((voidptr*)(b.$field.name))) {') fn_builder.write_string('*((voidptr*)(a.$field.name)) == *((voidptr*)(b.$field.name))')
} else { } else {
fn_builder.writeln('\tif (a.$field.name != b.$field.name) {') fn_builder.write_string('a.$field.name == b.$field.name')
} }
fn_builder.writeln('\t\treturn false;')
fn_builder.writeln('\t}')
} }
fn_builder.writeln('\treturn true;') } else {
fn_builder.write_string('true')
}
fn_builder.writeln(';')
fn_builder.writeln('}') fn_builder.writeln('}')
return ptr_typ return ptr_styp
} }
fn (mut g Gen) gen_alias_equality_fn(left ast.Type) string { fn (mut g Gen) gen_alias_equality_fn(left_type ast.Type) string {
ptr_typ := g.typ(left).trim('*') left := g.unwrap(left_type)
if ptr_typ in g.alias_fn_definitions { ptr_styp := g.typ(left.typ.set_nr_muls(0))
return ptr_typ if ptr_styp in g.alias_fn_definitions {
return ptr_styp
} }
g.alias_fn_definitions << ptr_typ g.alias_fn_definitions << ptr_styp
left_sym := g.table.get_type_symbol(left) info := left.sym.info as ast.Alias
info := left_sym.info as ast.Alias g.type_definitions.writeln('static bool ${ptr_styp}_alias_eq($ptr_styp a, $ptr_styp b); // auto')
g.type_definitions.writeln('static bool ${ptr_typ}_alias_eq($ptr_typ a, $ptr_typ b); // auto')
mut fn_builder := strings.new_builder(512) mut fn_builder := strings.new_builder(512)
fn_builder.writeln('static bool ${ptr_typ}_alias_eq($ptr_typ a, $ptr_typ b) {') fn_builder.writeln('static bool ${ptr_styp}_alias_eq($ptr_styp a, $ptr_styp b) {')
sym := g.table.get_type_symbol(info.parent_type) sym := g.table.get_type_symbol(info.parent_type)
if sym.kind == .string { if sym.kind == .string {
fn_builder.writeln('\tif (!string__eq(a, b)) {') fn_builder.writeln('\treturn string__eq(a, b);')
} else if sym.kind == .sum_type && !left.is_ptr() { } else if sym.kind == .sum_type && !left.typ.is_ptr() {
eq_fn := g.gen_sumtype_equality_fn(info.parent_type) eq_fn := g.gen_sumtype_equality_fn(info.parent_type)
fn_builder.writeln('\tif (!${eq_fn}_sumtype_eq(a, b)) {') fn_builder.writeln('\treturn ${eq_fn}_sumtype_eq(a, b);')
} else if sym.kind == .struct_ && !left.is_ptr() { } else if sym.kind == .struct_ && !left.typ.is_ptr() {
eq_fn := g.gen_struct_equality_fn(info.parent_type) eq_fn := g.gen_struct_equality_fn(info.parent_type)
fn_builder.writeln('\tif (!${eq_fn}_struct_eq(a, b)) {') fn_builder.writeln('\treturn ${eq_fn}_struct_eq(a, b);')
} else if sym.kind == .array && !left.is_ptr() { } else if sym.kind == .array && !left.typ.is_ptr() {
eq_fn := g.gen_array_equality_fn(info.parent_type) eq_fn := g.gen_array_equality_fn(info.parent_type)
fn_builder.writeln('\tif (!${eq_fn}_arr_eq(a, b)) {') fn_builder.writeln('\treturn ${eq_fn}_arr_eq(a, b);')
} else if sym.kind == .array_fixed && !left.is_ptr() { } else if sym.kind == .array_fixed && !left.typ.is_ptr() {
eq_fn := g.gen_fixed_array_equality_fn(info.parent_type) eq_fn := g.gen_fixed_array_equality_fn(info.parent_type)
fn_builder.writeln('\tif (!${eq_fn}_arr_eq(a, b)) {') fn_builder.writeln('\treturn ${eq_fn}_arr_eq(a, b);')
} else if sym.kind == .map && !left.is_ptr() { } else if sym.kind == .map && !left.typ.is_ptr() {
eq_fn := g.gen_map_equality_fn(info.parent_type) eq_fn := g.gen_map_equality_fn(info.parent_type)
fn_builder.writeln('\tif (!${eq_fn}_map_eq(a, b)) {') fn_builder.writeln('\treturn ${eq_fn}_map_eq(a, b);')
} else if sym.kind == .function { } else if sym.kind == .function {
fn_builder.writeln('\tif (*((voidptr*)(a)) != *((voidptr*)(b))) {') fn_builder.writeln('\treturn *((voidptr*)(a)) == *((voidptr*)(b));')
} else { } else {
fn_builder.writeln('\tif (a != b) {') fn_builder.writeln('\treturn a == b;')
} }
fn_builder.writeln('\t\treturn false;')
fn_builder.writeln('\t}')
fn_builder.writeln('\treturn true;')
fn_builder.writeln('}') fn_builder.writeln('}')
g.auto_fn_definitions << fn_builder.str() g.auto_fn_definitions << fn_builder.str()
return ptr_typ return ptr_styp
} }
fn (mut g Gen) gen_array_equality_fn(left ast.Type) string { fn (mut g Gen) gen_array_equality_fn(left_type ast.Type) string {
ptr_typ := g.typ(left).trim('*') left := g.unwrap(left_type)
if ptr_typ in g.array_fn_definitions { ptr_styp := g.typ(left.typ.set_nr_muls(0))
return ptr_typ if ptr_styp in g.array_fn_definitions {
return ptr_styp
} }
g.array_fn_definitions << ptr_typ g.array_fn_definitions << ptr_styp
left_sym := g.table.get_type_symbol(left) elem := g.unwrap(left.sym.array_info().elem_type)
elem_typ := left_sym.array_info().elem_type ptr_elem_styp := g.typ(elem.typ)
ptr_elem_typ := g.typ(elem_typ) g.type_definitions.writeln('static bool ${ptr_styp}_arr_eq($ptr_styp a, $ptr_styp b); // auto')
elem_sym := g.table.get_type_symbol(elem_typ)
g.type_definitions.writeln('static bool ${ptr_typ}_arr_eq($ptr_typ a, $ptr_typ b); // auto')
mut fn_builder := strings.new_builder(512) mut fn_builder := strings.new_builder(512)
fn_builder.writeln('static bool ${ptr_typ}_arr_eq($ptr_typ a, $ptr_typ b) {') fn_builder.writeln('static bool ${ptr_styp}_arr_eq($ptr_styp a, $ptr_styp b) {')
fn_builder.writeln('\tif (a.len != b.len) {') fn_builder.writeln('\tif (a.len != b.len) {')
fn_builder.writeln('\t\treturn false;') fn_builder.writeln('\t\treturn false;')
fn_builder.writeln('\t}') fn_builder.writeln('\t}')
fn_builder.writeln('\tfor (int i = 0; i < a.len; ++i) {') fn_builder.writeln('\tfor (int i = 0; i < a.len; ++i) {')
// compare every pair of elements of the two arrays // compare every pair of elements of the two arrays
if elem_sym.kind == .string { if elem.sym.kind == .string {
fn_builder.writeln('\t\tif (!string__eq(*(($ptr_elem_typ*)((byte*)a.data+(i*a.element_size))), *(($ptr_elem_typ*)((byte*)b.data+(i*b.element_size))))) {') fn_builder.writeln('\t\tif (!string__eq(*(($ptr_elem_styp*)((byte*)a.data+(i*a.element_size))), *(($ptr_elem_styp*)((byte*)b.data+(i*b.element_size))))) {')
} else if elem_sym.kind == .sum_type && !elem_typ.is_ptr() { } else if elem.sym.kind == .sum_type && !elem.typ.is_ptr() {
eq_fn := g.gen_sumtype_equality_fn(elem_typ) eq_fn := g.gen_sumtype_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq((($ptr_elem_typ*)a.data)[i], (($ptr_elem_typ*)b.data)[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq((($ptr_elem_styp*)a.data)[i], (($ptr_elem_styp*)b.data)[i])) {')
} else if elem_sym.kind == .struct_ && !elem_typ.is_ptr() { } else if elem.sym.kind == .struct_ && !elem.typ.is_ptr() {
eq_fn := g.gen_struct_equality_fn(elem_typ) eq_fn := g.gen_struct_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq((($ptr_elem_typ*)a.data)[i], (($ptr_elem_typ*)b.data)[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq((($ptr_elem_styp*)a.data)[i], (($ptr_elem_styp*)b.data)[i])) {')
} else if elem_sym.kind == .array && !elem_typ.is_ptr() { } else if elem.sym.kind == .array && !elem.typ.is_ptr() {
eq_fn := g.gen_array_equality_fn(elem_typ) eq_fn := g.gen_array_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq((($ptr_elem_typ*)a.data)[i], (($ptr_elem_typ*)b.data)[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq((($ptr_elem_styp*)a.data)[i], (($ptr_elem_styp*)b.data)[i])) {')
} else if elem_sym.kind == .array_fixed && !elem_typ.is_ptr() { } else if elem.sym.kind == .array_fixed && !elem.typ.is_ptr() {
eq_fn := g.gen_fixed_array_equality_fn(elem_typ) eq_fn := g.gen_fixed_array_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq((($ptr_elem_typ*)a.data)[i], (($ptr_elem_typ*)b.data)[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq((($ptr_elem_styp*)a.data)[i], (($ptr_elem_styp*)b.data)[i])) {')
} else if elem_sym.kind == .map && !elem_typ.is_ptr() { } else if elem.sym.kind == .map && !elem.typ.is_ptr() {
eq_fn := g.gen_map_equality_fn(elem_typ) eq_fn := g.gen_map_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_map_eq((($ptr_elem_typ*)a.data)[i], (($ptr_elem_typ*)b.data)[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_map_eq((($ptr_elem_styp*)a.data)[i], (($ptr_elem_styp*)b.data)[i])) {')
} else if elem_sym.kind == .alias && !elem_typ.is_ptr() { } else if elem.sym.kind == .alias && !elem.typ.is_ptr() {
eq_fn := g.gen_alias_equality_fn(elem_typ) eq_fn := g.gen_alias_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq((($ptr_elem_typ*)a.data)[i], (($ptr_elem_typ*)b.data)[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq((($ptr_elem_styp*)a.data)[i], (($ptr_elem_styp*)b.data)[i])) {')
} else if elem_sym.kind == .function { } else if elem.sym.kind == .function {
fn_builder.writeln('\t\tif (*((voidptr*)((byte*)a.data+(i*a.element_size))) != *((voidptr*)((byte*)b.data+(i*b.element_size)))) {') fn_builder.writeln('\t\tif (*((voidptr*)((byte*)a.data+(i*a.element_size))) != *((voidptr*)((byte*)b.data+(i*b.element_size)))) {')
} else { } else {
fn_builder.writeln('\t\tif (*(($ptr_elem_typ*)((byte*)a.data+(i*a.element_size))) != *(($ptr_elem_typ*)((byte*)b.data+(i*b.element_size)))) {') fn_builder.writeln('\t\tif (*(($ptr_elem_styp*)((byte*)a.data+(i*a.element_size))) != *(($ptr_elem_styp*)((byte*)b.data+(i*b.element_size)))) {')
} }
fn_builder.writeln('\t\t\treturn false;') fn_builder.writeln('\t\t\treturn false;')
fn_builder.writeln('\t\t}') fn_builder.writeln('\t\t}')
@ -206,46 +208,46 @@ fn (mut g Gen) gen_array_equality_fn(left ast.Type) string {
fn_builder.writeln('\treturn true;') fn_builder.writeln('\treturn true;')
fn_builder.writeln('}') fn_builder.writeln('}')
g.auto_fn_definitions << fn_builder.str() g.auto_fn_definitions << fn_builder.str()
return ptr_typ return ptr_styp
} }
fn (mut g Gen) gen_fixed_array_equality_fn(left ast.Type) string { fn (mut g Gen) gen_fixed_array_equality_fn(left_type ast.Type) string {
ptr_typ := g.typ(left).trim('*') left := g.unwrap(left_type)
if ptr_typ in g.array_fn_definitions { ptr_styp := g.typ(left.typ.set_nr_muls(0))
return ptr_typ if ptr_styp in g.array_fn_definitions {
return ptr_styp
} }
g.array_fn_definitions << ptr_typ g.array_fn_definitions << ptr_styp
left_sym := g.table.get_type_symbol(left) elem_info := left.sym.array_fixed_info()
elem_info := left_sym.array_fixed_info() elem := g.unwrap(elem_info.elem_type)
elem_typ := elem_info.elem_type
size := elem_info.size size := elem_info.size
elem_sym := g.table.get_type_symbol(elem_typ) g.type_definitions.writeln('static bool ${ptr_styp}_arr_eq($ptr_styp a, $ptr_styp b); // auto')
g.type_definitions.writeln('static bool ${ptr_typ}_arr_eq($ptr_typ a, $ptr_typ b); // auto')
mut fn_builder := strings.new_builder(512) mut fn_builder := strings.new_builder(512)
fn_builder.writeln('static bool ${ptr_typ}_arr_eq($ptr_typ a, $ptr_typ b) {') fn_builder.writeln('static bool ${ptr_styp}_arr_eq($ptr_styp a, $ptr_styp b) {')
fn_builder.writeln('\tfor (int i = 0; i < $size; ++i) {') fn_builder.writeln('\tfor (int i = 0; i < $size; ++i) {')
// compare every pair of elements of the two fixed arrays // compare every pair of elements of the two fixed arrays
if elem_sym.kind == .string { if elem.sym.kind == .string {
fn_builder.writeln('\t\tif (!string__eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!string__eq(a[i], b[i])) {')
} else if elem_sym.kind == .sum_type && !elem_typ.is_ptr() { } else if elem.sym.kind == .sum_type && !elem.typ.is_ptr() {
eq_fn := g.gen_sumtype_equality_fn(elem_typ) eq_fn := g.gen_sumtype_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq(a[i], b[i])) {')
} else if elem_sym.kind == .struct_ && !elem_typ.is_ptr() { } else if elem.sym.kind == .struct_ && !elem.typ.is_ptr() {
eq_fn := g.gen_struct_equality_fn(elem_typ) eq_fn := g.gen_struct_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq(a[i], b[i])) {')
} else if elem_sym.kind == .array && !elem_typ.is_ptr() { } else if elem.sym.kind == .array && !elem.typ.is_ptr() {
eq_fn := g.gen_array_equality_fn(elem_typ) eq_fn := g.gen_array_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(a[i], b[i])) {')
} else if elem_sym.kind == .array_fixed && !elem_typ.is_ptr() { } else if elem.sym.kind == .array_fixed && !elem.typ.is_ptr() {
eq_fn := g.gen_fixed_array_equality_fn(elem_typ) eq_fn := g.gen_fixed_array_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(a[i], b[i])) {')
} else if elem_sym.kind == .map && !elem_typ.is_ptr() { } else if elem.sym.kind == .map && !elem.typ.is_ptr() {
eq_fn := g.gen_map_equality_fn(elem_typ) eq_fn := g.gen_map_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_map_eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_map_eq(a[i], b[i])) {')
} else if elem_sym.kind == .alias && !elem_typ.is_ptr() { } else if elem.sym.kind == .alias && !elem.typ.is_ptr() {
eq_fn := g.gen_alias_equality_fn(elem_typ) eq_fn := g.gen_alias_equality_fn(elem.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq(a[i], b[i])) {') fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq(a[i], b[i])) {')
} else if elem_sym.kind == .function { } else if elem.sym.kind == .function {
fn_builder.writeln('\t\tif (a[i] != b[i]) {') fn_builder.writeln('\t\tif (a[i] != b[i]) {')
} else { } else {
fn_builder.writeln('\t\tif (a[i] != b[i]) {') fn_builder.writeln('\t\tif (a[i] != b[i]) {')
@ -256,21 +258,22 @@ fn (mut g Gen) gen_fixed_array_equality_fn(left ast.Type) string {
fn_builder.writeln('\treturn true;') fn_builder.writeln('\treturn true;')
fn_builder.writeln('}') fn_builder.writeln('}')
g.auto_fn_definitions << fn_builder.str() g.auto_fn_definitions << fn_builder.str()
return ptr_typ return ptr_styp
} }
fn (mut g Gen) gen_map_equality_fn(left ast.Type) string { fn (mut g Gen) gen_map_equality_fn(left_type ast.Type) string {
ptr_typ := g.typ(left).trim('*') left := g.unwrap(left_type)
if ptr_typ in g.map_fn_definitions { ptr_styp := g.typ(left.typ.set_nr_muls(0))
return ptr_typ if ptr_styp in g.map_fn_definitions {
return ptr_styp
} }
g.map_fn_definitions << ptr_typ g.map_fn_definitions << ptr_styp
left_sym := g.table.get_type_symbol(left) value := g.unwrap(left.sym.map_info().value_type)
value_typ := left_sym.map_info().value_type ptr_value_styp := g.typ(value.typ)
ptr_value_typ := g.typ(value_typ) g.type_definitions.writeln('static bool ${ptr_styp}_map_eq($ptr_styp a, $ptr_styp b); // auto')
g.type_definitions.writeln('static bool ${ptr_typ}_map_eq($ptr_typ a, $ptr_typ b); // auto')
mut fn_builder := strings.new_builder(512) mut fn_builder := strings.new_builder(512)
fn_builder.writeln('static bool ${ptr_typ}_map_eq($ptr_typ a, $ptr_typ b) {') fn_builder.writeln('static bool ${ptr_styp}_map_eq($ptr_styp a, $ptr_styp b) {')
fn_builder.writeln('\tif (a.len != b.len) {') fn_builder.writeln('\tif (a.len != b.len) {')
fn_builder.writeln('\t\treturn false;') fn_builder.writeln('\t\treturn false;')
fn_builder.writeln('\t}') fn_builder.writeln('\t}')
@ -278,10 +281,9 @@ fn (mut g Gen) gen_map_equality_fn(left ast.Type) string {
fn_builder.writeln('\t\tif (!DenseArray_has_index(&a.key_values, i)) continue;') fn_builder.writeln('\t\tif (!DenseArray_has_index(&a.key_values, i)) continue;')
fn_builder.writeln('\t\tvoidptr k = DenseArray_key(&a.key_values, i);') fn_builder.writeln('\t\tvoidptr k = DenseArray_key(&a.key_values, i);')
fn_builder.writeln('\t\tif (!map_exists(&b, k)) return false;') fn_builder.writeln('\t\tif (!map_exists(&b, k)) return false;')
kind := g.table.type_kind(value_typ) kind := g.table.type_kind(value.typ)
if kind == .function { if kind == .function {
value_sym := g.table.get_type_symbol(value_typ) func := value.sym.info as ast.FnType
func := value_sym.info as ast.FnType
ret_styp := g.typ(func.func.return_type) ret_styp := g.typ(func.func.return_type)
fn_builder.write_string('\t\t$ret_styp (*v) (') fn_builder.write_string('\t\t$ret_styp (*v) (')
arg_len := func.func.params.len arg_len := func.func.params.len
@ -294,41 +296,41 @@ fn (mut g Gen) gen_map_equality_fn(left ast.Type) string {
} }
fn_builder.writeln(') = *(voidptr*)map_get(&a, k, &(voidptr[]){ 0 });') fn_builder.writeln(') = *(voidptr*)map_get(&a, k, &(voidptr[]){ 0 });')
} else { } else {
fn_builder.writeln('\t\t$ptr_value_typ v = *($ptr_value_typ*)map_get(&a, k, &($ptr_value_typ[]){ 0 });') fn_builder.writeln('\t\t$ptr_value_styp v = *($ptr_value_styp*)map_get(&a, k, &($ptr_value_styp[]){ 0 });')
} }
match kind { match kind {
.string { .string {
fn_builder.writeln('\t\tif (!fast_string_eq(*(string*)map_get(&b, k, &(string[]){_SLIT("")}), v)) {') fn_builder.writeln('\t\tif (!fast_string_eq(*(string*)map_get(&b, k, &(string[]){_SLIT("")}), v)) {')
} }
.sum_type { .sum_type {
eq_fn := g.gen_sumtype_equality_fn(value_typ) eq_fn := g.gen_sumtype_equality_fn(value.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq(*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }), v)) {') fn_builder.writeln('\t\tif (!${eq_fn}_sumtype_eq(*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }), v)) {')
} }
.struct_ { .struct_ {
eq_fn := g.gen_struct_equality_fn(value_typ) eq_fn := g.gen_struct_equality_fn(value.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq(*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }), v)) {') fn_builder.writeln('\t\tif (!${eq_fn}_struct_eq(*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }), v)) {')
} }
.array { .array {
eq_fn := g.gen_array_equality_fn(value_typ) eq_fn := g.gen_array_equality_fn(value.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }), v)) {') fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }), v)) {')
} }
.array_fixed { .array_fixed {
eq_fn := g.gen_fixed_array_equality_fn(value_typ) eq_fn := g.gen_fixed_array_equality_fn(value.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }), v)) {') fn_builder.writeln('\t\tif (!${eq_fn}_arr_eq(*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }), v)) {')
} }
.map { .map {
eq_fn := g.gen_map_equality_fn(value_typ) eq_fn := g.gen_map_equality_fn(value.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_map_eq(*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }), v)) {') fn_builder.writeln('\t\tif (!${eq_fn}_map_eq(*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }), v)) {')
} }
.alias { .alias {
eq_fn := g.gen_alias_equality_fn(value_typ) eq_fn := g.gen_alias_equality_fn(value.typ)
fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq(*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }), v)) {') fn_builder.writeln('\t\tif (!${eq_fn}_alias_eq(*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }), v)) {')
} }
.function { .function {
fn_builder.writeln('\t\tif (*(voidptr*)map_get(&b, k, &(voidptr[]){ 0 }) != v) {') fn_builder.writeln('\t\tif (*(voidptr*)map_get(&b, k, &(voidptr[]){ 0 }) != v) {')
} }
else { else {
fn_builder.writeln('\t\tif (*($ptr_value_typ*)map_get(&b, k, &($ptr_value_typ[]){ 0 }) != v) {') fn_builder.writeln('\t\tif (*($ptr_value_styp*)map_get(&b, k, &($ptr_value_styp[]){ 0 }) != v) {')
} }
} }
fn_builder.writeln('\t\t\treturn false;') fn_builder.writeln('\t\t\treturn false;')
@ -337,5 +339,5 @@ fn (mut g Gen) gen_map_equality_fn(left ast.Type) string {
fn_builder.writeln('\treturn true;') fn_builder.writeln('\treturn true;')
fn_builder.writeln('}') fn_builder.writeln('}')
g.auto_fn_definitions << fn_builder.str() g.auto_fn_definitions << fn_builder.str()
return ptr_typ return ptr_styp
} }