Compare commits
	
		
			10 Commits
		
	
	
		
			d5cbb5b6c8
			...
			6d7305f5f0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 6d7305f5f0 | ||
|  | 4dc7652aa2 | ||
|  | 0b1fc7eb6a | ||
|  | 7487308948 | ||
|  | 0baae921ed | ||
|  | a8c81522cd | ||
|  | a16fb19797 | ||
|  | 7e19a88ad2 | ||
|  | 0e57f694be | ||
|  | ecb18949df | 
| @ -16,7 +16,6 @@ | |||||||
|             .hash = "122041f6531ee2cd10e7d5f81817c50b45037affc95d748cbcd71a766866fb6030d4", |             .hash = "122041f6531ee2cd10e7d5f81817c50b45037affc95d748cbcd71a766866fb6030d4", | ||||||
|         }, |         }, | ||||||
|         .binned_allocator = .{ |         .binned_allocator = .{ | ||||||
|             // upstream: https://gist.github.com/silversquirl/c1e4840048fdf48e669b6eac76d80634 |  | ||||||
|             .url = "https://gist.github.com/FalsePattern/48fded613c115e16e91c46db8642c7e4/archive/75e3d5e6a0e0cf23dbf7abfe16831e23c38721bc.tar.gz", |             .url = "https://gist.github.com/FalsePattern/48fded613c115e16e91c46db8642c7e4/archive/75e3d5e6a0e0cf23dbf7abfe16831e23c38721bc.tar.gz", | ||||||
|             .hash = "1220ba896ddd4258eed9274b36284d4cc4600ee69c4c0978cbe237ec09a524a2e252", |             .hash = "1220ba896ddd4258eed9274b36284d4cc4600ee69c4c0978cbe237ec09a524a2e252", | ||||||
|         }, |         }, | ||||||
|  | |||||||
| @ -32,32 +32,27 @@ pub fn receiveMessage(client: *Client) !InMessage.Header { | |||||||
|     const Header = InMessage.Header; |     const Header = InMessage.Header; | ||||||
|     const fifo = client.pooler.fifo(.in); |     const fifo = client.pooler.fifo(.in); | ||||||
| 
 | 
 | ||||||
|     while (try client.pooler.poll()) { |     var first_run = true; | ||||||
|         const buf = fifo.readableSlice(0); |     var header: ?Header = null; | ||||||
|         assert(fifo.readableLength() == buf.len); |     while (first_run or try client.pooler.poll()) { | ||||||
|         if (buf.len >= @sizeOf(Header)) { |         first_run = false; | ||||||
|             // workaround for https://github.com/ziglang/zig/issues/14904 | 
 | ||||||
|  |         if (header == null) { | ||||||
|  |             if (fifo.readableLength() < @sizeOf(Header)) continue; | ||||||
|  |             const buf = fifo.readableSlice(0); | ||||||
|             const bytes_len = bswap_and_workaround_u32(buf[4..][0..4]); |             const bytes_len = bswap_and_workaround_u32(buf[4..][0..4]); | ||||||
|             const tag = bswap_and_workaround_tag(buf[0..][0..4]); |             const tag = bswap_and_workaround_tag(buf[0..][0..4]); | ||||||
| 
 |             header = Header{ | ||||||
|             if (buf.len - @sizeOf(Header) >= bytes_len) { |                 .tag = tag, | ||||||
|                 fifo.discard(@sizeOf(Header)); |                 .bytes_len = bytes_len, | ||||||
|                 return .{ |             }; | ||||||
|                     .tag = tag, |             fifo.discard(@sizeOf(Header)); | ||||||
|                     .bytes_len = bytes_len, |  | ||||||
|                 }; |  | ||||||
|             } else { |  | ||||||
|                 const needed = bytes_len - (buf.len - @sizeOf(Header)); |  | ||||||
|                 const write_buffer = try fifo.writableWithSize(needed); |  | ||||||
|                 const amt = try client.in.readAll(write_buffer); |  | ||||||
|                 fifo.update(amt); |  | ||||||
|                 continue; |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         const write_buffer = try fifo.writableWithSize(256); |         if (header) |h| { | ||||||
|         const amt = try client.in.read(write_buffer); |             if (fifo.readableLength() < h.bytes_len) continue; | ||||||
|         fifo.update(amt); |             return h; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|     return error.Timeout; |     return error.Timeout; | ||||||
| } | } | ||||||
|  | |||||||
| @ -338,7 +338,7 @@ pub const Key = union(enum) { | |||||||
|                 .anyerror => .ErrorSet, |                 .anyerror => .ErrorSet, | ||||||
|                 .noreturn => .NoReturn, |                 .noreturn => .NoReturn, | ||||||
|                 .anyframe_type => .AnyFrame, |                 .anyframe_type => .AnyFrame, | ||||||
|                 .empty_struct_literal => .Struct, |                 .empty_struct_type => .Struct, | ||||||
|                 .null_type => .Null, |                 .null_type => .Null, | ||||||
|                 .undefined_type => .Undefined, |                 .undefined_type => .Undefined, | ||||||
|                 .enum_literal_type => .EnumLiteral, |                 .enum_literal_type => .EnumLiteral, | ||||||
| @ -634,7 +634,7 @@ pub const Key = union(enum) { | |||||||
|                 .enum_literal_type, |                 .enum_literal_type, | ||||||
|                 => Index.none, |                 => Index.none, | ||||||
| 
 | 
 | ||||||
|                 .empty_struct_literal => Index.empty_aggregate, |                 .empty_struct_type => Index.empty_aggregate, | ||||||
|                 .void => Index.void_value, |                 .void => Index.void_value, | ||||||
|                 .noreturn => Index.unreachable_value, |                 .noreturn => Index.unreachable_value, | ||||||
|                 .null_type => Index.null_value, |                 .null_type => Index.null_value, | ||||||
| @ -659,8 +659,8 @@ pub const Key = union(enum) { | |||||||
|             .int_type => |int_info| { |             .int_type => |int_info| { | ||||||
|                 if (int_info.bits == 0) { |                 if (int_info.bits == 0) { | ||||||
|                     switch (int_info.signedness) { |                     switch (int_info.signedness) { | ||||||
|                         .unsigned => return Index.zero, |                         .unsigned => return Index.zero_comptime_int, | ||||||
|                         .signed => return Index.zero, // do we need a signed zero? |                         .signed => return Index.zero_comptime_int, // do we need a signed zero? | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|                 return Index.none; |                 return Index.none; | ||||||
| @ -835,7 +835,7 @@ pub const Key = union(enum) { | |||||||
| 
 | 
 | ||||||
|                 .null_type => try writer.writeAll("@TypeOf(null)"), |                 .null_type => try writer.writeAll("@TypeOf(null)"), | ||||||
|                 .undefined_type => try writer.writeAll("@TypeOf(undefined)"), |                 .undefined_type => try writer.writeAll("@TypeOf(undefined)"), | ||||||
|                 .empty_struct_literal => try writer.writeAll("@TypeOf(.{})"), |                 .empty_struct_type => try writer.writeAll("@TypeOf(.{})"), | ||||||
|                 .enum_literal_type => try writer.writeAll("@TypeOf(.enum_literal)"), |                 .enum_literal_type => try writer.writeAll("@TypeOf(.enum_literal)"), | ||||||
| 
 | 
 | ||||||
|                 .atomic_order => try writer.writeAll("std.builtin.AtomicOrder"), |                 .atomic_order => try writer.writeAll("std.builtin.AtomicOrder"), | ||||||
| @ -1104,7 +1104,7 @@ pub const Index = enum(u32) { | |||||||
|     comptime_float_type, |     comptime_float_type, | ||||||
|     noreturn_type, |     noreturn_type, | ||||||
|     anyframe_type, |     anyframe_type, | ||||||
|     empty_struct_literal, |     empty_struct_type, | ||||||
|     null_type, |     null_type, | ||||||
|     undefined_type, |     undefined_type, | ||||||
|     enum_literal_type, |     enum_literal_type, | ||||||
| @ -1114,19 +1114,22 @@ pub const Index = enum(u32) { | |||||||
|     address_space_type, |     address_space_type, | ||||||
|     float_mode_type, |     float_mode_type, | ||||||
|     reduce_op_type, |     reduce_op_type, | ||||||
|     modifier_type, |     call_modifier_type, | ||||||
|     prefetch_options_type, |     prefetch_options_type, | ||||||
|     export_options_type, |     export_options_type, | ||||||
|     extern_options_type, |     extern_options_type, | ||||||
|     type_info_type, |     type_info_type, | ||||||
|     manyptr_u8_type, |     manyptr_u8_type, | ||||||
|     manyptr_const_u8_type, |     manyptr_const_u8_type, | ||||||
|  |     manyptr_const_u8_sentinel_0_type, | ||||||
|     fn_noreturn_no_args_type, |     fn_noreturn_no_args_type, | ||||||
|     fn_void_no_args_type, |     fn_void_no_args_type, | ||||||
|     fn_naked_noreturn_no_args_type, |     fn_naked_noreturn_no_args_type, | ||||||
|     fn_ccc_void_no_args_type, |     fn_ccc_void_no_args_type, | ||||||
|     single_const_pointer_to_comptime_int_type, |     single_const_pointer_to_comptime_int_type, | ||||||
|     const_slice_u8_type, |     slice_const_u8_type, | ||||||
|  |     slice_const_u8_sentinel_0_type, | ||||||
|  |     optional_noreturn_type, | ||||||
|     anyerror_void_error_union_type, |     anyerror_void_error_union_type, | ||||||
|     generic_poison_type, |     generic_poison_type, | ||||||
|     unknown_type, |     unknown_type, | ||||||
| @ -1134,9 +1137,17 @@ pub const Index = enum(u32) { | |||||||
|     /// `undefined` (untyped) |     /// `undefined` (untyped) | ||||||
|     undefined_value, |     undefined_value, | ||||||
|     /// `0` (comptime_int) |     /// `0` (comptime_int) | ||||||
|     zero, |     zero_comptime_int, | ||||||
|  |     /// `0` (u8) | ||||||
|  |     zero_u8, | ||||||
|  |     /// `0` (usize) | ||||||
|  |     zero_usize, | ||||||
|     /// `1` (comptime_int) |     /// `1` (comptime_int) | ||||||
|     one, |     one_comptime_int, | ||||||
|  |     /// `1` (u8) | ||||||
|  |     one_u8, | ||||||
|  |     /// `1` (usize) | ||||||
|  |     one_usize, | ||||||
|     /// `{}` |     /// `{}` | ||||||
|     void_value, |     void_value, | ||||||
|     /// `unreachable` (noreturn type) |     /// `unreachable` (noreturn type) | ||||||
| @ -1149,10 +1160,6 @@ pub const Index = enum(u32) { | |||||||
|     bool_false, |     bool_false, | ||||||
|     /// `.{}` (untyped) |     /// `.{}` (untyped) | ||||||
|     empty_aggregate, |     empty_aggregate, | ||||||
|     /// `0` (usize) |  | ||||||
|     zero_usize, |  | ||||||
|     /// `1` (usize) |  | ||||||
|     one_usize, |  | ||||||
|     the_only_possible_value, |     the_only_possible_value, | ||||||
|     generic_poison, |     generic_poison, | ||||||
|     // unknown value of unknown type |     // unknown value of unknown type | ||||||
| @ -1345,7 +1352,7 @@ pub const SimpleType = enum(u32) { | |||||||
|     comptime_float, |     comptime_float, | ||||||
|     noreturn, |     noreturn, | ||||||
|     anyframe_type, |     anyframe_type, | ||||||
|     empty_struct_literal, |     empty_struct_type, | ||||||
|     null_type, |     null_type, | ||||||
|     undefined_type, |     undefined_type, | ||||||
|     enum_literal_type, |     enum_literal_type, | ||||||
| @ -1424,7 +1431,7 @@ pub fn init(gpa: Allocator) Allocator.Error!InternPool { | |||||||
|         .{ .index = .comptime_float_type, .key = .{ .simple_type = .comptime_float } }, |         .{ .index = .comptime_float_type, .key = .{ .simple_type = .comptime_float } }, | ||||||
|         .{ .index = .noreturn_type, .key = .{ .simple_type = .noreturn } }, |         .{ .index = .noreturn_type, .key = .{ .simple_type = .noreturn } }, | ||||||
|         .{ .index = .anyframe_type, .key = .{ .simple_type = .anyframe_type } }, |         .{ .index = .anyframe_type, .key = .{ .simple_type = .anyframe_type } }, | ||||||
|         .{ .index = .empty_struct_literal, .key = .{ .simple_type = .empty_struct_literal } }, |         .{ .index = .empty_struct_type, .key = .{ .simple_type = .empty_struct_type } }, | ||||||
|         .{ .index = .null_type, .key = .{ .simple_type = .null_type } }, |         .{ .index = .null_type, .key = .{ .simple_type = .null_type } }, | ||||||
|         .{ .index = .undefined_type, .key = .{ .simple_type = .undefined_type } }, |         .{ .index = .undefined_type, .key = .{ .simple_type = .undefined_type } }, | ||||||
|         .{ .index = .enum_literal_type, .key = .{ .simple_type = .enum_literal_type } }, |         .{ .index = .enum_literal_type, .key = .{ .simple_type = .enum_literal_type } }, | ||||||
| @ -1435,41 +1442,45 @@ pub fn init(gpa: Allocator) Allocator.Error!InternPool { | |||||||
|         .{ .index = .address_space_type, .key = .{ .simple_type = .address_space } }, |         .{ .index = .address_space_type, .key = .{ .simple_type = .address_space } }, | ||||||
|         .{ .index = .float_mode_type, .key = .{ .simple_type = .float_mode } }, |         .{ .index = .float_mode_type, .key = .{ .simple_type = .float_mode } }, | ||||||
|         .{ .index = .reduce_op_type, .key = .{ .simple_type = .reduce_op } }, |         .{ .index = .reduce_op_type, .key = .{ .simple_type = .reduce_op } }, | ||||||
|         .{ .index = .modifier_type, .key = .{ .simple_type = .modifier } }, |         .{ .index = .call_modifier_type, .key = .{ .simple_type = .modifier } }, | ||||||
|         .{ .index = .prefetch_options_type, .key = .{ .simple_type = .prefetch_options } }, |         .{ .index = .prefetch_options_type, .key = .{ .simple_type = .prefetch_options } }, | ||||||
|         .{ .index = .export_options_type, .key = .{ .simple_type = .export_options } }, |         .{ .index = .export_options_type, .key = .{ .simple_type = .export_options } }, | ||||||
|         .{ .index = .extern_options_type, .key = .{ .simple_type = .extern_options } }, |         .{ .index = .extern_options_type, .key = .{ .simple_type = .extern_options } }, | ||||||
|         .{ .index = .type_info_type, .key = .{ .simple_type = .type_info } }, |         .{ .index = .type_info_type, .key = .{ .simple_type = .type_info } }, | ||||||
|         .{ .index = .manyptr_u8_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .size = .Many } } }, |         .{ .index = .manyptr_u8_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .size = .Many } } }, | ||||||
|         .{ .index = .manyptr_const_u8_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .size = .Many, .is_const = true } } }, |         .{ .index = .manyptr_const_u8_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .size = .Many, .is_const = true } } }, | ||||||
|  |         .{ .index = .manyptr_const_u8_sentinel_0_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .sentinel = .zero_u8, .size = .Many, .is_const = true } } }, | ||||||
|         .{ .index = .fn_noreturn_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .noreturn_type } } }, |         .{ .index = .fn_noreturn_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .noreturn_type } } }, | ||||||
|         .{ .index = .fn_void_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .void_type } } }, |         .{ .index = .fn_void_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .void_type } } }, | ||||||
|         .{ .index = .fn_naked_noreturn_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .void_type, .calling_convention = .Naked } } }, |         .{ .index = .fn_naked_noreturn_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .void_type, .calling_convention = .Naked } } }, | ||||||
|         .{ .index = .fn_ccc_void_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .void_type, .calling_convention = .C } } }, |         .{ .index = .fn_ccc_void_no_args_type, .key = .{ .function_type = .{ .args = &.{}, .return_type = .void_type, .calling_convention = .C } } }, | ||||||
|         .{ .index = .single_const_pointer_to_comptime_int_type, .key = .{ .pointer_type = .{ .elem_type = .comptime_int_type, .size = .One, .is_const = true } } }, |         .{ .index = .single_const_pointer_to_comptime_int_type, .key = .{ .pointer_type = .{ .elem_type = .comptime_int_type, .size = .One, .is_const = true } } }, | ||||||
|         .{ .index = .const_slice_u8_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .size = .Slice, .is_const = true } } }, |         .{ .index = .slice_const_u8_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .size = .Slice, .is_const = true } } }, | ||||||
|  |         .{ .index = .slice_const_u8_sentinel_0_type, .key = .{ .pointer_type = .{ .elem_type = .u8_type, .sentinel = .zero_u8, .size = .Slice, .is_const = true } } }, | ||||||
|  |         .{ .index = .optional_noreturn_type, .key = .{ .optional_type = .{ .payload_type = .noreturn_type } } }, | ||||||
|         .{ .index = .anyerror_void_error_union_type, .key = .{ .error_union_type = .{ .error_set_type = .anyerror_type, .payload_type = .void_type } } }, |         .{ .index = .anyerror_void_error_union_type, .key = .{ .error_union_type = .{ .error_set_type = .anyerror_type, .payload_type = .void_type } } }, | ||||||
|         .{ .index = .generic_poison_type, .key = .{ .simple_type = .generic_poison } }, |         .{ .index = .generic_poison_type, .key = .{ .simple_type = .generic_poison } }, | ||||||
|         .{ .index = .unknown_type, .key = .{ .simple_type = .unknown } }, |         .{ .index = .unknown_type, .key = .{ .simple_type = .unknown } }, | ||||||
| 
 | 
 | ||||||
|         .{ .index = .undefined_value, .key = .{ .simple_value = .undefined_value } }, |         .{ .index = .undefined_value, .key = .{ .simple_value = .undefined_value } }, | ||||||
|         .{ .index = .zero, .key = .{ .int_u64_value = .{ .ty = .comptime_int_type, .int = 0 } } }, |         .{ .index = .zero_comptime_int, .key = .{ .int_u64_value = .{ .ty = .comptime_int_type, .int = 0 } } }, | ||||||
|         .{ .index = .one, .key = .{ .int_u64_value = .{ .ty = .comptime_int_type, .int = 1 } } }, |         .{ .index = .zero_u8, .key = .{ .int_u64_value = .{ .ty = .u8_type, .int = 0 } } }, | ||||||
|  |         .{ .index = .zero_usize, .key = .{ .int_u64_value = .{ .ty = .usize_type, .int = 0 } } }, | ||||||
|  |         .{ .index = .one_comptime_int, .key = .{ .int_u64_value = .{ .ty = .comptime_int_type, .int = 1 } } }, | ||||||
|  |         .{ .index = .one_u8, .key = .{ .int_u64_value = .{ .ty = .u8_type, .int = 1 } } }, | ||||||
|  |         .{ .index = .one_usize, .key = .{ .int_u64_value = .{ .ty = .usize_type, .int = 1 } } }, | ||||||
|         .{ .index = .void_value, .key = .{ .simple_value = .void_value } }, |         .{ .index = .void_value, .key = .{ .simple_value = .void_value } }, | ||||||
|         .{ .index = .unreachable_value, .key = .{ .simple_value = .unreachable_value } }, |         .{ .index = .unreachable_value, .key = .{ .simple_value = .unreachable_value } }, | ||||||
|         .{ .index = .null_value, .key = .{ .simple_value = .null_value } }, |         .{ .index = .null_value, .key = .{ .simple_value = .null_value } }, | ||||||
|         .{ .index = .bool_true, .key = .{ .simple_value = .bool_true } }, |         .{ .index = .bool_true, .key = .{ .simple_value = .bool_true } }, | ||||||
|         .{ .index = .bool_false, .key = .{ .simple_value = .bool_false } }, |         .{ .index = .bool_false, .key = .{ .simple_value = .bool_false } }, | ||||||
| 
 |         .{ .index = .empty_aggregate, .key = .{ .aggregate = .{ .ty = .empty_struct_type, .values = &.{} } } }, | ||||||
|         .{ .index = .empty_aggregate, .key = .{ .aggregate = .{ .ty = .empty_struct_literal, .values = &.{} } } }, |  | ||||||
|         .{ .index = .zero_usize, .key = .{ .int_u64_value = .{ .ty = .usize_type, .int = 0 } } }, |  | ||||||
|         .{ .index = .one_usize, .key = .{ .int_u64_value = .{ .ty = .usize_type, .int = 1 } } }, |  | ||||||
|         .{ .index = .the_only_possible_value, .key = .{ .simple_value = .the_only_possible_value } }, |         .{ .index = .the_only_possible_value, .key = .{ .simple_value = .the_only_possible_value } }, | ||||||
|         .{ .index = .generic_poison, .key = .{ .simple_value = .generic_poison } }, |         .{ .index = .generic_poison, .key = .{ .simple_value = .generic_poison } }, | ||||||
|         .{ .index = .unknown_unknown, .key = .{ .unknown_value = .{ .ty = .unknown_type } } }, |         .{ .index = .unknown_unknown, .key = .{ .unknown_value = .{ .ty = .unknown_type } } }, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     const extra_count = 4 * @sizeOf(Pointer) + @sizeOf(ErrorUnion) + 4 * @sizeOf(Function) + 4 * @sizeOf(InternPool.U64Value); |     const extra_count = 6 * @sizeOf(Pointer) + @sizeOf(ErrorUnion) + 4 * @sizeOf(Function) + 6 * @sizeOf(InternPool.U64Value); | ||||||
| 
 | 
 | ||||||
|     try ip.map.ensureTotalCapacity(gpa, items.len); |     try ip.map.ensureTotalCapacity(gpa, items.len); | ||||||
|     try ip.items.ensureTotalCapacity(gpa, items.len); |     try ip.items.ensureTotalCapacity(gpa, items.len); | ||||||
| @ -3107,7 +3118,7 @@ test "pointer type" { | |||||||
|     const @"[*:0]u32" = try ip.get(gpa, .{ .pointer_type = .{ |     const @"[*:0]u32" = try ip.get(gpa, .{ .pointer_type = .{ | ||||||
|         .elem_type = .u32_type, |         .elem_type = .u32_type, | ||||||
|         .size = .Many, |         .size = .Many, | ||||||
|         .sentinel = .zero, |         .sentinel = .zero_comptime_int, | ||||||
|     } }); |     } }); | ||||||
|     const @"[]u32" = try ip.get(gpa, .{ .pointer_type = .{ |     const @"[]u32" = try ip.get(gpa, .{ .pointer_type = .{ | ||||||
|         .elem_type = .u32_type, |         .elem_type = .u32_type, | ||||||
| @ -3116,7 +3127,7 @@ test "pointer type" { | |||||||
|     const @"[:0]u32" = try ip.get(gpa, .{ .pointer_type = .{ |     const @"[:0]u32" = try ip.get(gpa, .{ .pointer_type = .{ | ||||||
|         .elem_type = .u32_type, |         .elem_type = .u32_type, | ||||||
|         .size = .Slice, |         .size = .Slice, | ||||||
|         .sentinel = .zero, |         .sentinel = .zero_comptime_int, | ||||||
|     } }); |     } }); | ||||||
|     const @"[*c]u32" = try ip.get(gpa, .{ .pointer_type = .{ |     const @"[*c]u32" = try ip.get(gpa, .{ .pointer_type = .{ | ||||||
|         .elem_type = .u32_type, |         .elem_type = .u32_type, | ||||||
| @ -3233,7 +3244,7 @@ test "array type" { | |||||||
|     const u32_0_0_array_type = try ip.get(gpa, .{ .array_type = .{ |     const u32_0_0_array_type = try ip.get(gpa, .{ .array_type = .{ | ||||||
|         .len = 3, |         .len = 3, | ||||||
|         .child = .u32_type, |         .child = .u32_type, | ||||||
|         .sentinel = .zero, |         .sentinel = .zero_comptime_int, | ||||||
|     } }); |     } }); | ||||||
| 
 | 
 | ||||||
|     try expect(i32_3_array_type != u32_0_0_array_type); |     try expect(i32_3_array_type != u32_0_0_array_type); | ||||||
|  | |||||||
| @ -989,16 +989,22 @@ fn resolveTypeOfNodeUncached(analyser: *Analyser, node_handle: NodeWithHandle) e | |||||||
| 
 | 
 | ||||||
|             const cast_map = std.ComptimeStringMap(void, .{ |             const cast_map = std.ComptimeStringMap(void, .{ | ||||||
|                 .{"@as"}, |                 .{"@as"}, | ||||||
|  |                 .{"@atomicLoad"}, | ||||||
|  |                 .{"@atomicRmw"}, | ||||||
|  |                 .{"@atomicStore"}, | ||||||
|                 .{"@bitCast"}, |                 .{"@bitCast"}, | ||||||
|                 .{"@fieldParentPtr"}, |                 .{"@mulAdd"}, | ||||||
|  |                 .{"@errSetCast"}, | ||||||
|  |                 .{"@fieldParentPtr"}, // the return type is actually a pointer | ||||||
|                 .{"@floatCast"}, |                 .{"@floatCast"}, | ||||||
|                 .{"@floatToInt"}, |                 .{"@intFromFloat"}, | ||||||
|                 .{"@intCast"}, |                 .{"@intCast"}, | ||||||
|                 .{"@intToEnum"}, |                 .{"@enumFromInt"}, | ||||||
|                 .{"@intToFloat"}, |                 .{"@floatFromInt"}, | ||||||
|                 .{"@intToPtr"}, |                 .{"@ptrFromInt"}, | ||||||
|                 .{"@truncate"}, |  | ||||||
|                 .{"@ptrCast"}, |                 .{"@ptrCast"}, | ||||||
|  |                 .{"@truncate"}, | ||||||
|  |                 .{"@unionInit"}, | ||||||
|             }); |             }); | ||||||
|             if (cast_map.has(call_name)) { |             if (cast_map.has(call_name)) { | ||||||
|                 if (params.len < 1) return null; |                 if (params.len < 1) return null; | ||||||
| @ -1134,6 +1140,33 @@ fn resolveTypeOfNodeUncached(analyser: *Analyser, node_handle: NodeWithHandle) e | |||||||
|                 .handle = handle, |                 .handle = handle, | ||||||
|             }; |             }; | ||||||
|         }, |         }, | ||||||
|  |         .block, | ||||||
|  |         .block_semicolon, | ||||||
|  |         .block_two, | ||||||
|  |         .block_two_semicolon, | ||||||
|  |         => { | ||||||
|  |             const first_token = tree.firstToken(node); | ||||||
|  |             if (token_tags[first_token] != .identifier) return null; | ||||||
|  | 
 | ||||||
|  |             const block_label = tree.tokenSlice(first_token); | ||||||
|  | 
 | ||||||
|  |             var buffer: [2]Ast.Node.Index = undefined; | ||||||
|  |             const statements = ast.blockStatements(tree, node, &buffer).?; | ||||||
|  | 
 | ||||||
|  |             for (statements) |child_idx| { | ||||||
|  |                 // TODO: Recursively find matching `break :label` (e.g. inside `if`) | ||||||
|  |                 if (node_tags[child_idx] == .@"break") { | ||||||
|  |                     if (datas[child_idx].lhs == 0) continue; | ||||||
|  |                     if (datas[child_idx].rhs == 0) continue; | ||||||
|  | 
 | ||||||
|  |                     const break_label = tree.tokenSlice(datas[child_idx].lhs); | ||||||
|  |                     if (!std.mem.eql(u8, block_label, break_label)) continue; | ||||||
|  | 
 | ||||||
|  |                     const operand = .{ .node = datas[child_idx].rhs, .handle = handle }; | ||||||
|  |                     return try analyser.resolveTypeOfNodeInternal(operand); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|         else => {}, |         else => {}, | ||||||
|     } |     } | ||||||
|     return null; |     return null; | ||||||
|  | |||||||
| @ -84,24 +84,31 @@ fn writeCallHint(builder: *Builder, call: Ast.full.Call, decl_handle: Analyser.D | |||||||
|     var buffer: [1]Ast.Node.Index = undefined; |     var buffer: [1]Ast.Node.Index = undefined; | ||||||
|     const fn_proto = decl_tree.fullFnProto(&buffer, fn_node) orelse return; |     const fn_proto = decl_tree.fullFnProto(&buffer, fn_node) orelse return; | ||||||
| 
 | 
 | ||||||
|     var i: usize = 0; |     var params = try std.ArrayListUnmanaged(Ast.full.FnProto.Param).initCapacity(builder.arena, fn_proto.ast.params.len); | ||||||
|     var it = fn_proto.iterate(&decl_tree); |     defer params.deinit(builder.arena); | ||||||
| 
 | 
 | ||||||
|     if (try builder.analyser.hasSelfParam(decl_handle.handle, fn_proto)) { |     var it = fn_proto.iterate(&decl_tree); | ||||||
|         _ = ast.nextFnParam(&it); |     while (ast.nextFnParam(&it)) |param| { | ||||||
|  |         try params.append(builder.arena, param); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     while (ast.nextFnParam(&it)) |param| : (i += 1) { |     const has_self_param = tree.tokens.items(.tag)[call.ast.lparen - 2] == .period and | ||||||
|         if (i >= call.ast.params.len) break; |         call.ast.params.len + 1 == params.items.len and | ||||||
|  |         try builder.analyser.hasSelfParam(decl_handle.handle, fn_proto); | ||||||
|  | 
 | ||||||
|  |     const parameters = params.items[@intFromBool(has_self_param)..]; | ||||||
|  |     const arguments = call.ast.params; | ||||||
|  |     const min_len = @min(parameters.len, arguments.len); | ||||||
|  |     for (parameters[0..min_len], call.ast.params[0..min_len]) |param, arg| { | ||||||
|         const name_token = param.name_token orelse continue; |         const name_token = param.name_token orelse continue; | ||||||
|         const name = decl_tree.tokenSlice(name_token); |         const name = decl_tree.tokenSlice(name_token); | ||||||
| 
 | 
 | ||||||
|         if (builder.config.inlay_hints_hide_redundant_param_names or builder.config.inlay_hints_hide_redundant_param_names_last_token) { |         if (builder.config.inlay_hints_hide_redundant_param_names or builder.config.inlay_hints_hide_redundant_param_names_last_token) { | ||||||
|             const last_param_token = tree.lastToken(call.ast.params[i]); |             const last_arg_token = tree.lastToken(arg); | ||||||
|             const param_name = tree.tokenSlice(last_param_token); |             const arg_name = tree.tokenSlice(last_arg_token); | ||||||
| 
 | 
 | ||||||
|             if (std.mem.eql(u8, param_name, name)) { |             if (std.mem.eql(u8, arg_name, name)) { | ||||||
|                 if (tree.firstToken(call.ast.params[i]) == last_param_token) { |                 if (tree.firstToken(arg) == last_arg_token) { | ||||||
|                     if (builder.config.inlay_hints_hide_redundant_param_names) |                     if (builder.config.inlay_hints_hide_redundant_param_names) | ||||||
|                         continue; |                         continue; | ||||||
|                 } else { |                 } else { | ||||||
| @ -122,7 +129,7 @@ fn writeCallHint(builder: *Builder, call: Ast.full.Call, decl_handle: Analyser.D | |||||||
|             offsets.nodeToSlice(decl_tree, param.type_expr); |             offsets.nodeToSlice(decl_tree, param.type_expr); | ||||||
| 
 | 
 | ||||||
|         try builder.appendParameterHint( |         try builder.appendParameterHint( | ||||||
|             tree.firstToken(call.ast.params[i]), |             tree.firstToken(arg), | ||||||
|             name, |             name, | ||||||
|             tooltip, |             tooltip, | ||||||
|             no_alias, |             no_alias, | ||||||
|  | |||||||
| @ -856,10 +856,35 @@ fn expr(gz: *GenZir, scope: *Scope, ri: ResultInfo, node: Ast.Node.Index) InnerE | |||||||
|             return rvalue(gz, ri, result, node); |             return rvalue(gz, ri, result, node); | ||||||
|         }, |         }, | ||||||
|         .slice => { |         .slice => { | ||||||
|  |             const extra = tree.extraData(node_datas[node].rhs, Ast.Node.Slice); | ||||||
|  |             const lhs_node = node_datas[node].lhs; | ||||||
|  |             const lhs_tag = node_tags[lhs_node]; | ||||||
|  |             const lhs_is_slice_sentinel = lhs_tag == .slice_sentinel; | ||||||
|  |             const lhs_is_open_slice = lhs_tag == .slice_open or | ||||||
|  |                 (lhs_is_slice_sentinel and tree.extraData(node_datas[lhs_node].rhs, Ast.Node.SliceSentinel).end == 0); | ||||||
|  |             if (lhs_is_open_slice and nodeIsTriviallyZero(tree, extra.start)) { | ||||||
|  |                 const lhs = try expr(gz, scope, .{ .rl = .ref }, node_datas[lhs_node].lhs); | ||||||
|  | 
 | ||||||
|  |                 const start = if (lhs_is_slice_sentinel) start: { | ||||||
|  |                     const lhs_extra = tree.extraData(node_datas[lhs_node].rhs, Ast.Node.SliceSentinel); | ||||||
|  |                     break :start try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, lhs_extra.start); | ||||||
|  |                 } else try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, node_datas[lhs_node].rhs); | ||||||
|  | 
 | ||||||
|  |                 const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); | ||||||
|  |                 const len = if (extra.end != 0) try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.end) else .none; | ||||||
|  |                 try emitDbgStmt(gz, cursor); | ||||||
|  |                 const result = try gz.addPlNode(.slice_length, node, Zir.Inst.SliceLength{ | ||||||
|  |                     .lhs = lhs, | ||||||
|  |                     .start = start, | ||||||
|  |                     .len = len, | ||||||
|  |                     .start_src_node_offset = gz.nodeIndexToRelative(lhs_node), | ||||||
|  |                     .sentinel = .none, | ||||||
|  |                 }); | ||||||
|  |                 return rvalue(gz, ri, result, node); | ||||||
|  |             } | ||||||
|             const lhs = try expr(gz, scope, .{ .rl = .ref }, node_datas[node].lhs); |             const lhs = try expr(gz, scope, .{ .rl = .ref }, node_datas[node].lhs); | ||||||
| 
 | 
 | ||||||
|             const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); |             const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); | ||||||
|             const extra = tree.extraData(node_datas[node].rhs, Ast.Node.Slice); |  | ||||||
|             const start = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.start); |             const start = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.start); | ||||||
|             const end = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.end); |             const end = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.end); | ||||||
|             try emitDbgStmt(gz, cursor); |             try emitDbgStmt(gz, cursor); | ||||||
| @ -871,10 +896,36 @@ fn expr(gz: *GenZir, scope: *Scope, ri: ResultInfo, node: Ast.Node.Index) InnerE | |||||||
|             return rvalue(gz, ri, result, node); |             return rvalue(gz, ri, result, node); | ||||||
|         }, |         }, | ||||||
|         .slice_sentinel => { |         .slice_sentinel => { | ||||||
|  |             const extra = tree.extraData(node_datas[node].rhs, Ast.Node.SliceSentinel); | ||||||
|  |             const lhs_node = node_datas[node].lhs; | ||||||
|  |             const lhs_tag = node_tags[lhs_node]; | ||||||
|  |             const lhs_is_slice_sentinel = lhs_tag == .slice_sentinel; | ||||||
|  |             const lhs_is_open_slice = lhs_tag == .slice_open or | ||||||
|  |                 (lhs_is_slice_sentinel and tree.extraData(node_datas[lhs_node].rhs, Ast.Node.SliceSentinel).end == 0); | ||||||
|  |             if (lhs_is_open_slice and nodeIsTriviallyZero(tree, extra.start)) { | ||||||
|  |                 const lhs = try expr(gz, scope, .{ .rl = .ref }, node_datas[lhs_node].lhs); | ||||||
|  | 
 | ||||||
|  |                 const start = if (lhs_is_slice_sentinel) start: { | ||||||
|  |                     const lhs_extra = tree.extraData(node_datas[lhs_node].rhs, Ast.Node.SliceSentinel); | ||||||
|  |                     break :start try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, lhs_extra.start); | ||||||
|  |                 } else try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, node_datas[lhs_node].rhs); | ||||||
|  | 
 | ||||||
|  |                 const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); | ||||||
|  |                 const len = if (extra.end != 0) try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.end) else .none; | ||||||
|  |                 const sentinel = try expr(gz, scope, .{ .rl = .none }, extra.sentinel); | ||||||
|  |                 try emitDbgStmt(gz, cursor); | ||||||
|  |                 const result = try gz.addPlNode(.slice_length, node, Zir.Inst.SliceLength{ | ||||||
|  |                     .lhs = lhs, | ||||||
|  |                     .start = start, | ||||||
|  |                     .len = len, | ||||||
|  |                     .start_src_node_offset = gz.nodeIndexToRelative(lhs_node), | ||||||
|  |                     .sentinel = sentinel, | ||||||
|  |                 }); | ||||||
|  |                 return rvalue(gz, ri, result, node); | ||||||
|  |             } | ||||||
|             const lhs = try expr(gz, scope, .{ .rl = .ref }, node_datas[node].lhs); |             const lhs = try expr(gz, scope, .{ .rl = .ref }, node_datas[node].lhs); | ||||||
| 
 | 
 | ||||||
|             const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); |             const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); | ||||||
|             const extra = tree.extraData(node_datas[node].rhs, Ast.Node.SliceSentinel); |  | ||||||
|             const start = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.start); |             const start = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.start); | ||||||
|             const end = if (extra.end != 0) try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.end) else .none; |             const end = if (extra.end != 0) try expr(gz, scope, .{ .rl = .{ .coerced_ty = .usize_type } }, extra.end) else .none; | ||||||
|             const sentinel = try expr(gz, scope, .{ .rl = .none }, extra.sentinel); |             const sentinel = try expr(gz, scope, .{ .rl = .none }, extra.sentinel); | ||||||
| @ -1579,7 +1630,7 @@ fn structInitExpr( | |||||||
| 
 | 
 | ||||||
|     if (struct_init.ast.type_expr == 0) { |     if (struct_init.ast.type_expr == 0) { | ||||||
|         if (struct_init.ast.fields.len == 0) { |         if (struct_init.ast.fields.len == 0) { | ||||||
|             return rvalue(gz, ri, .empty_struct, node); |             return rvalue(gz, ri, .empty_struct_type, node); | ||||||
|         } |         } | ||||||
|     } else array: { |     } else array: { | ||||||
|         const node_tags = tree.nodes.items(.tag); |         const node_tags = tree.nodes.items(.tag); | ||||||
| @ -2438,7 +2489,7 @@ fn addEnsureResult(gz: *GenZir, maybe_unused_result: Zir.Inst.Ref, statement: As | |||||||
|         switch (zir_tags[inst]) { |         switch (zir_tags[inst]) { | ||||||
|             // For some instructions, modify the zir data |             // For some instructions, modify the zir data | ||||||
|             // so we can avoid a separate ensure_result_used instruction. |             // so we can avoid a separate ensure_result_used instruction. | ||||||
|             .call => { |             .call, .field_call => { | ||||||
|                 const extra_index = gz.astgen.instructions.items(.data)[inst].pl_node.payload_index; |                 const extra_index = gz.astgen.instructions.items(.data)[inst].pl_node.payload_index; | ||||||
|                 const slot = &gz.astgen.extra.items[extra_index]; |                 const slot = &gz.astgen.extra.items[extra_index]; | ||||||
|                 var flags = @bitCast(Zir.Inst.Call.Flags, slot.*); |                 var flags = @bitCast(Zir.Inst.Call.Flags, slot.*); | ||||||
| @ -2513,7 +2564,6 @@ fn addEnsureResult(gz: *GenZir, maybe_unused_result: Zir.Inst.Ref, statement: As | |||||||
|             .field_ptr, |             .field_ptr, | ||||||
|             .field_ptr_init, |             .field_ptr_init, | ||||||
|             .field_val, |             .field_val, | ||||||
|             .field_call_bind, |  | ||||||
|             .field_ptr_named, |             .field_ptr_named, | ||||||
|             .field_val_named, |             .field_val_named, | ||||||
|             .func, |             .func, | ||||||
| @ -2564,15 +2614,10 @@ fn addEnsureResult(gz: *GenZir, maybe_unused_result: Zir.Inst.Ref, statement: As | |||||||
|             .slice_start, |             .slice_start, | ||||||
|             .slice_end, |             .slice_end, | ||||||
|             .slice_sentinel, |             .slice_sentinel, | ||||||
|  |             .slice_length, | ||||||
|             .import, |             .import, | ||||||
|             .switch_block, |             .switch_block, | ||||||
|             .switch_cond, |             .switch_block_ref, | ||||||
|             .switch_cond_ref, |  | ||||||
|             .switch_capture, |  | ||||||
|             .switch_capture_ref, |  | ||||||
|             .switch_capture_multi, |  | ||||||
|             .switch_capture_multi_ref, |  | ||||||
|             .switch_capture_tag, |  | ||||||
|             .struct_init_empty, |             .struct_init_empty, | ||||||
|             .struct_init, |             .struct_init, | ||||||
|             .struct_init_ref, |             .struct_init_ref, | ||||||
| @ -2588,15 +2633,15 @@ fn addEnsureResult(gz: *GenZir, maybe_unused_result: Zir.Inst.Ref, statement: As | |||||||
|             .error_set_decl, |             .error_set_decl, | ||||||
|             .error_set_decl_anon, |             .error_set_decl_anon, | ||||||
|             .error_set_decl_func, |             .error_set_decl_func, | ||||||
|             .int_to_enum, |             .enum_from_int, | ||||||
|             .enum_to_int, |             .int_from_enum, | ||||||
|             .type_info, |             .type_info, | ||||||
|             .size_of, |             .size_of, | ||||||
|             .bit_size_of, |             .bit_size_of, | ||||||
|             .typeof_log2_int_type, |             .typeof_log2_int_type, | ||||||
|             .ptr_to_int, |             .int_from_ptr, | ||||||
|             .align_of, |             .align_of, | ||||||
|             .bool_to_int, |             .int_from_bool, | ||||||
|             .embed_file, |             .embed_file, | ||||||
|             .error_name, |             .error_name, | ||||||
|             .sqrt, |             .sqrt, | ||||||
| @ -2617,9 +2662,9 @@ fn addEnsureResult(gz: *GenZir, maybe_unused_result: Zir.Inst.Ref, statement: As | |||||||
|             .type_name, |             .type_name, | ||||||
|             .frame_type, |             .frame_type, | ||||||
|             .frame_size, |             .frame_size, | ||||||
|             .float_to_int, |             .int_from_float, | ||||||
|             .int_to_float, |             .float_from_int, | ||||||
|             .int_to_ptr, |             .ptr_from_int, | ||||||
|             .float_cast, |             .float_cast, | ||||||
|             .int_cast, |             .int_cast, | ||||||
|             .ptr_cast, |             .ptr_cast, | ||||||
| @ -2916,7 +2961,7 @@ fn deferStmt( | |||||||
|         try gz.astgen.instructions.append(gz.astgen.gpa, .{ |         try gz.astgen.instructions.append(gz.astgen.gpa, .{ | ||||||
|             .tag = .extended, |             .tag = .extended, | ||||||
|             .data = .{ .extended = .{ |             .data = .{ .extended = .{ | ||||||
|                 .opcode = .errdefer_err_code, |                 .opcode = .value_placeholder, | ||||||
|                 .small = undefined, |                 .small = undefined, | ||||||
|                 .operand = undefined, |                 .operand = undefined, | ||||||
|             } }, |             } }, | ||||||
| @ -2938,11 +2983,27 @@ fn deferStmt( | |||||||
|     if (have_err_code) try gz.addDbgBlockEnd(); |     if (have_err_code) try gz.addDbgBlockEnd(); | ||||||
|     _ = try defer_gen.addBreak(.break_inline, 0, .void_value); |     _ = try defer_gen.addBreak(.break_inline, 0, .void_value); | ||||||
| 
 | 
 | ||||||
|  |     // We must handle ref_table for remapped_err_code manually. | ||||||
|     const body = defer_gen.instructionsSlice(); |     const body = defer_gen.instructionsSlice(); | ||||||
|     const body_len = gz.astgen.countBodyLenAfterFixups(body); |     const body_len = blk: { | ||||||
|  |         var refs: u32 = 0; | ||||||
|  |         if (have_err_code) { | ||||||
|  |             var cur_inst = remapped_err_code; | ||||||
|  |             while (gz.astgen.ref_table.get(cur_inst)) |ref_inst| { | ||||||
|  |                 refs += 1; | ||||||
|  |                 cur_inst = ref_inst; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         break :blk gz.astgen.countBodyLenAfterFixups(body) + refs; | ||||||
|  |     }; | ||||||
| 
 | 
 | ||||||
|     const index = @intCast(u32, gz.astgen.extra.items.len); |     const index = @intCast(u32, gz.astgen.extra.items.len); | ||||||
|     try gz.astgen.extra.ensureUnusedCapacity(gz.astgen.gpa, body_len); |     try gz.astgen.extra.ensureUnusedCapacity(gz.astgen.gpa, body_len); | ||||||
|  |     if (have_err_code) { | ||||||
|  |         if (gz.astgen.ref_table.fetchRemove(remapped_err_code)) |kv| { | ||||||
|  |             gz.astgen.appendPossiblyRefdBodyInst(&gz.astgen.extra, kv.value); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     gz.astgen.appendBodyWithFixups(body); |     gz.astgen.appendBodyWithFixups(body); | ||||||
| 
 | 
 | ||||||
|     const defer_scope = try block_arena.create(Scope.Defer); |     const defer_scope = try block_arena.create(Scope.Defer); | ||||||
| @ -3874,7 +3935,7 @@ fn fnDecl( | |||||||
|     var section_gz = decl_gz.makeSubBlock(params_scope); |     var section_gz = decl_gz.makeSubBlock(params_scope); | ||||||
|     defer section_gz.unstack(); |     defer section_gz.unstack(); | ||||||
|     const section_ref: Zir.Inst.Ref = if (fn_proto.ast.section_expr == 0) .none else inst: { |     const section_ref: Zir.Inst.Ref = if (fn_proto.ast.section_expr == 0) .none else inst: { | ||||||
|         const inst = try expr(&decl_gz, params_scope, .{ .rl = .{ .coerced_ty = .const_slice_u8_type } }, fn_proto.ast.section_expr); |         const inst = try expr(&decl_gz, params_scope, .{ .rl = .{ .coerced_ty = .slice_const_u8_type } }, fn_proto.ast.section_expr); | ||||||
|         if (section_gz.instructionsSlice().len == 0) { |         if (section_gz.instructionsSlice().len == 0) { | ||||||
|             // In this case we will send a len=0 body which can be encoded more efficiently. |             // In this case we will send a len=0 body which can be encoded more efficiently. | ||||||
|             break :inst inst; |             break :inst inst; | ||||||
| @ -3908,9 +3969,9 @@ fn fnDecl( | |||||||
|             break :blk inst; |             break :blk inst; | ||||||
|         } else if (is_extern) { |         } else if (is_extern) { | ||||||
|             // note: https://github.com/ziglang/zig/issues/5269 |             // note: https://github.com/ziglang/zig/issues/5269 | ||||||
|             break :blk .calling_convention_c; |             break :blk .unknown_unknown; // TODO calling_convention_c | ||||||
|         } else if (has_inline_keyword) { |         } else if (has_inline_keyword) { | ||||||
|             break :blk .calling_convention_inline; |             break :blk .unknown_unknown; // calling_convention_inline | ||||||
|         } else { |         } else { | ||||||
|             break :blk .none; |             break :blk .none; | ||||||
|         } |         } | ||||||
| @ -4077,7 +4138,7 @@ fn globalVarDecl( | |||||||
|         break :inst try expr(&block_scope, &block_scope.base, .{ .rl = .{ .ty = .address_space_type } }, var_decl.ast.addrspace_node); |         break :inst try expr(&block_scope, &block_scope.base, .{ .rl = .{ .ty = .address_space_type } }, var_decl.ast.addrspace_node); | ||||||
|     }; |     }; | ||||||
|     const section_inst: Zir.Inst.Ref = if (var_decl.ast.section_node == 0) .none else inst: { |     const section_inst: Zir.Inst.Ref = if (var_decl.ast.section_node == 0) .none else inst: { | ||||||
|         break :inst try comptimeExpr(&block_scope, &block_scope.base, .{ .rl = .{ .ty = .const_slice_u8_type } }, var_decl.ast.section_node); |         break :inst try comptimeExpr(&block_scope, &block_scope.base, .{ .rl = .{ .ty = .slice_const_u8_type } }, var_decl.ast.section_node); | ||||||
|     }; |     }; | ||||||
|     const has_section_or_addrspace = section_inst != .none or addrspace_inst != .none; |     const has_section_or_addrspace = section_inst != .none or addrspace_inst != .none; | ||||||
|     wip_members.nextDecl(is_pub, is_export, align_inst != .none, has_section_or_addrspace); |     wip_members.nextDecl(is_pub, is_export, align_inst != .none, has_section_or_addrspace); | ||||||
| @ -4437,7 +4498,7 @@ fn testDecl( | |||||||
|         .cc_gz = null, |         .cc_gz = null, | ||||||
|         .align_ref = .none, |         .align_ref = .none, | ||||||
|         .align_gz = null, |         .align_gz = null, | ||||||
|         .ret_ref = .void_type, |         .ret_ref = .anyerror_void_error_union_type, | ||||||
|         .ret_gz = null, |         .ret_gz = null, | ||||||
|         .section_ref = .none, |         .section_ref = .none, | ||||||
|         .section_gz = null, |         .section_gz = null, | ||||||
| @ -4450,7 +4511,7 @@ fn testDecl( | |||||||
|         .body_gz = &fn_block, |         .body_gz = &fn_block, | ||||||
|         .lib_name = 0, |         .lib_name = 0, | ||||||
|         .is_var_args = false, |         .is_var_args = false, | ||||||
|         .is_inferred_error = true, |         .is_inferred_error = false, | ||||||
|         .is_test = true, |         .is_test = true, | ||||||
|         .is_extern = false, |         .is_extern = false, | ||||||
|         .is_noinline = false, |         .is_noinline = false, | ||||||
| @ -6655,6 +6716,7 @@ fn switchExpr( | |||||||
|     // for the following variables, make note of the special prong AST node index, |     // for the following variables, make note of the special prong AST node index, | ||||||
|     // and bail out with a compile error if there are multiple special prongs present. |     // and bail out with a compile error if there are multiple special prongs present. | ||||||
|     var any_payload_is_ref = false; |     var any_payload_is_ref = false; | ||||||
|  |     var any_has_tag_capture = false; | ||||||
|     var scalar_cases_len: u32 = 0; |     var scalar_cases_len: u32 = 0; | ||||||
|     var multi_cases_len: u32 = 0; |     var multi_cases_len: u32 = 0; | ||||||
|     var inline_cases_len: u32 = 0; |     var inline_cases_len: u32 = 0; | ||||||
| @ -6665,8 +6727,12 @@ fn switchExpr( | |||||||
|     for (case_nodes) |case_node| { |     for (case_nodes) |case_node| { | ||||||
|         const case = tree.fullSwitchCase(case_node).?; |         const case = tree.fullSwitchCase(case_node).?; | ||||||
|         if (case.payload_token) |payload_token| { |         if (case.payload_token) |payload_token| { | ||||||
|             if (token_tags[payload_token] == .asterisk) { |             const ident = if (token_tags[payload_token] == .asterisk) blk: { | ||||||
|                 any_payload_is_ref = true; |                 any_payload_is_ref = true; | ||||||
|  |                 break :blk payload_token + 1; | ||||||
|  |             } else payload_token; | ||||||
|  |             if (token_tags[ident + 1] == .comma) { | ||||||
|  |                 any_has_tag_capture = true; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         // Check for else/`_` prong. |         // Check for else/`_` prong. | ||||||
| @ -6775,13 +6841,7 @@ fn switchExpr( | |||||||
|     const operand_lc = LineColumn{ astgen.source_line - parent_gz.decl_line, astgen.source_column }; |     const operand_lc = LineColumn{ astgen.source_line - parent_gz.decl_line, astgen.source_column }; | ||||||
| 
 | 
 | ||||||
|     const raw_operand = try expr(parent_gz, scope, operand_ri, operand_node); |     const raw_operand = try expr(parent_gz, scope, operand_ri, operand_node); | ||||||
|     const cond_tag: Zir.Inst.Tag = if (any_payload_is_ref) .switch_cond_ref else .switch_cond; |     const item_ri: ResultInfo = .{ .rl = .none }; | ||||||
|     const cond = try parent_gz.addUnNode(cond_tag, raw_operand, operand_node); |  | ||||||
|     // Sema expects a dbg_stmt immediately after switch_cond(_ref) |  | ||||||
|     try emitDbgStmt(parent_gz, operand_lc); |  | ||||||
|     // We need the type of the operand to use as the result location for all the prong items. |  | ||||||
|     const cond_ty_inst = try parent_gz.addUnNode(.typeof, cond, operand_node); |  | ||||||
|     const item_ri: ResultInfo = .{ .rl = .{ .ty = cond_ty_inst } }; |  | ||||||
| 
 | 
 | ||||||
|     // This contains the data that goes into the `extra` array for the SwitchBlock/SwitchBlockMulti, |     // This contains the data that goes into the `extra` array for the SwitchBlock/SwitchBlockMulti, | ||||||
|     // except the first cases_nodes.len slots are a table that indexes payloads later in the array, with |     // except the first cases_nodes.len slots are a table that indexes payloads later in the array, with | ||||||
| @ -6800,13 +6860,30 @@ fn switchExpr( | |||||||
|     block_scope.instructions_top = GenZir.unstacked_top; |     block_scope.instructions_top = GenZir.unstacked_top; | ||||||
|     block_scope.setBreakResultInfo(ri); |     block_scope.setBreakResultInfo(ri); | ||||||
| 
 | 
 | ||||||
|  |     // Sema expects a dbg_stmt immediately before switch_block(_ref) | ||||||
|  |     try emitDbgStmt(parent_gz, operand_lc); | ||||||
|     // This gets added to the parent block later, after the item expressions. |     // This gets added to the parent block later, after the item expressions. | ||||||
|     const switch_block = try parent_gz.makeBlockInst(.switch_block, switch_node); |     const switch_tag: Zir.Inst.Tag = if (any_payload_is_ref) .switch_block_ref else .switch_block; | ||||||
|  |     const switch_block = try parent_gz.makeBlockInst(switch_tag, switch_node); | ||||||
| 
 | 
 | ||||||
|     // We re-use this same scope for all cases, including the special prong, if any. |     // We re-use this same scope for all cases, including the special prong, if any. | ||||||
|     var case_scope = parent_gz.makeSubBlock(&block_scope.base); |     var case_scope = parent_gz.makeSubBlock(&block_scope.base); | ||||||
|     case_scope.instructions_top = GenZir.unstacked_top; |     case_scope.instructions_top = GenZir.unstacked_top; | ||||||
| 
 | 
 | ||||||
|  |     // If any prong has an inline tag capture, allocate a shared dummy instruction for it | ||||||
|  |     const tag_inst = if (any_has_tag_capture) tag_inst: { | ||||||
|  |         const inst = @intCast(Zir.Inst.Index, astgen.instructions.len); | ||||||
|  |         try astgen.instructions.append(astgen.gpa, .{ | ||||||
|  |             .tag = .extended, | ||||||
|  |             .data = .{ .extended = .{ | ||||||
|  |                 .opcode = .value_placeholder, | ||||||
|  |                 .small = undefined, | ||||||
|  |                 .operand = undefined, | ||||||
|  |             } }, // TODO rename opcode | ||||||
|  |         }); | ||||||
|  |         break :tag_inst inst; | ||||||
|  |     } else undefined; | ||||||
|  | 
 | ||||||
|     // In this pass we generate all the item and prong expressions. |     // In this pass we generate all the item and prong expressions. | ||||||
|     var multi_case_index: u32 = 0; |     var multi_case_index: u32 = 0; | ||||||
|     var scalar_case_index: u32 = 0; |     var scalar_case_index: u32 = 0; | ||||||
| @ -6820,17 +6897,22 @@ fn switchExpr( | |||||||
|         var dbg_var_inst: Zir.Inst.Ref = undefined; |         var dbg_var_inst: Zir.Inst.Ref = undefined; | ||||||
|         var dbg_var_tag_name: ?u32 = null; |         var dbg_var_tag_name: ?u32 = null; | ||||||
|         var dbg_var_tag_inst: Zir.Inst.Ref = undefined; |         var dbg_var_tag_inst: Zir.Inst.Ref = undefined; | ||||||
|         var capture_inst: Zir.Inst.Index = 0; |         var has_tag_capture = false; | ||||||
|         var tag_inst: Zir.Inst.Index = 0; |  | ||||||
|         var capture_val_scope: Scope.LocalVal = undefined; |         var capture_val_scope: Scope.LocalVal = undefined; | ||||||
|         var tag_scope: Scope.LocalVal = undefined; |         var tag_scope: Scope.LocalVal = undefined; | ||||||
|  | 
 | ||||||
|  |         var capture: Zir.Inst.SwitchBlock.ProngInfo.Capture = .none; | ||||||
|  | 
 | ||||||
|         const sub_scope = blk: { |         const sub_scope = blk: { | ||||||
|             const payload_token = case.payload_token orelse break :blk &case_scope.base; |             const payload_token = case.payload_token orelse break :blk &case_scope.base; | ||||||
|             const ident = if (token_tags[payload_token] == .asterisk) |             const ident = if (token_tags[payload_token] == .asterisk) | ||||||
|                 payload_token + 1 |                 payload_token + 1 | ||||||
|             else |             else | ||||||
|                 payload_token; |                 payload_token; | ||||||
|  | 
 | ||||||
|             const is_ptr = ident != payload_token; |             const is_ptr = ident != payload_token; | ||||||
|  |             capture = if (is_ptr) .by_ref else .by_val; | ||||||
|  | 
 | ||||||
|             const ident_slice = tree.tokenSlice(ident); |             const ident_slice = tree.tokenSlice(ident); | ||||||
|             var payload_sub_scope: *Scope = undefined; |             var payload_sub_scope: *Scope = undefined; | ||||||
|             if (mem.eql(u8, ident_slice, "_")) { |             if (mem.eql(u8, ident_slice, "_")) { | ||||||
| @ -6839,53 +6921,18 @@ fn switchExpr( | |||||||
|                 } |                 } | ||||||
|                 payload_sub_scope = &case_scope.base; |                 payload_sub_scope = &case_scope.base; | ||||||
|             } else { |             } else { | ||||||
|                 if (case_node == special_node) { |  | ||||||
|                     const capture_tag: Zir.Inst.Tag = if (is_ptr) |  | ||||||
|                         .switch_capture_ref |  | ||||||
|                     else |  | ||||||
|                         .switch_capture; |  | ||||||
|                     capture_inst = @intCast(Zir.Inst.Index, astgen.instructions.len); |  | ||||||
|                     try astgen.instructions.append(gpa, .{ |  | ||||||
|                         .tag = capture_tag, |  | ||||||
|                         .data = .{ |  | ||||||
|                             .switch_capture = .{ |  | ||||||
|                                 .switch_inst = switch_block, |  | ||||||
|                                 // Max int communicates that this is the else/underscore prong. |  | ||||||
|                                 .prong_index = std.math.maxInt(u32), |  | ||||||
|                             }, |  | ||||||
|                         }, |  | ||||||
|                     }); |  | ||||||
|                 } else { |  | ||||||
|                     const is_multi_case_bits: u2 = @intFromBool(is_multi_case); |  | ||||||
|                     const is_ptr_bits: u2 = @intFromBool(is_ptr); |  | ||||||
|                     const capture_tag: Zir.Inst.Tag = switch ((is_multi_case_bits << 1) | is_ptr_bits) { |  | ||||||
|                         0b00 => .switch_capture, |  | ||||||
|                         0b01 => .switch_capture_ref, |  | ||||||
|                         0b10 => .switch_capture_multi, |  | ||||||
|                         0b11 => .switch_capture_multi_ref, |  | ||||||
|                     }; |  | ||||||
|                     const capture_index = if (is_multi_case) multi_case_index else scalar_case_index; |  | ||||||
|                     capture_inst = @intCast(Zir.Inst.Index, astgen.instructions.len); |  | ||||||
|                     try astgen.instructions.append(gpa, .{ |  | ||||||
|                         .tag = capture_tag, |  | ||||||
|                         .data = .{ .switch_capture = .{ |  | ||||||
|                             .switch_inst = switch_block, |  | ||||||
|                             .prong_index = capture_index, |  | ||||||
|                         } }, |  | ||||||
|                     }); |  | ||||||
|                 } |  | ||||||
|                 const capture_name = try astgen.identAsString(ident); |                 const capture_name = try astgen.identAsString(ident); | ||||||
|                 try astgen.detectLocalShadowing(&case_scope.base, capture_name, ident, ident_slice, .capture); |                 try astgen.detectLocalShadowing(&case_scope.base, capture_name, ident, ident_slice, .capture); | ||||||
|                 capture_val_scope = .{ |                 capture_val_scope = .{ | ||||||
|                     .parent = &case_scope.base, |                     .parent = &case_scope.base, | ||||||
|                     .gen_zir = &case_scope, |                     .gen_zir = &case_scope, | ||||||
|                     .name = capture_name, |                     .name = capture_name, | ||||||
|                     .inst = indexToRef(capture_inst), |                     .inst = indexToRef(switch_block), | ||||||
|                     .token_src = payload_token, |                     .token_src = payload_token, | ||||||
|                     .id_cat = .capture, |                     .id_cat = .capture, | ||||||
|                 }; |                 }; | ||||||
|                 dbg_var_name = capture_name; |                 dbg_var_name = capture_name; | ||||||
|                 dbg_var_inst = indexToRef(capture_inst); |                 dbg_var_inst = indexToRef(switch_block); | ||||||
|                 payload_sub_scope = &capture_val_scope.base; |                 payload_sub_scope = &capture_val_scope.base; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
| @ -6901,14 +6948,9 @@ fn switchExpr( | |||||||
|             } |             } | ||||||
|             const tag_name = try astgen.identAsString(tag_token); |             const tag_name = try astgen.identAsString(tag_token); | ||||||
|             try astgen.detectLocalShadowing(payload_sub_scope, tag_name, tag_token, tag_slice, .@"switch tag capture"); |             try astgen.detectLocalShadowing(payload_sub_scope, tag_name, tag_token, tag_slice, .@"switch tag capture"); | ||||||
|             tag_inst = @intCast(Zir.Inst.Index, astgen.instructions.len); | 
 | ||||||
|             try astgen.instructions.append(gpa, .{ |             assert(any_has_tag_capture); | ||||||
|                 .tag = .switch_capture_tag, |             has_tag_capture = true; | ||||||
|                 .data = .{ .un_tok = .{ |  | ||||||
|                     .operand = cond, |  | ||||||
|                     .src_tok = case_scope.tokenIndexToRelative(tag_token), |  | ||||||
|                 } }, |  | ||||||
|             }); |  | ||||||
| 
 | 
 | ||||||
|             tag_scope = .{ |             tag_scope = .{ | ||||||
|                 .parent = payload_sub_scope, |                 .parent = payload_sub_scope, | ||||||
| @ -6974,8 +7016,6 @@ fn switchExpr( | |||||||
|             case_scope.instructions_top = parent_gz.instructions.items.len; |             case_scope.instructions_top = parent_gz.instructions.items.len; | ||||||
|             defer case_scope.unstack(); |             defer case_scope.unstack(); | ||||||
| 
 | 
 | ||||||
|             if (capture_inst != 0) try case_scope.instructions.append(gpa, capture_inst); |  | ||||||
|             if (tag_inst != 0) try case_scope.instructions.append(gpa, tag_inst); |  | ||||||
|             try case_scope.addDbgBlockBegin(); |             try case_scope.addDbgBlockBegin(); | ||||||
|             if (dbg_var_name) |some| { |             if (dbg_var_name) |some| { | ||||||
|                 try case_scope.addDbgVar(.dbg_var_val, some, dbg_var_inst); |                 try case_scope.addDbgVar(.dbg_var_val, some, dbg_var_inst); | ||||||
| @ -6993,10 +7033,42 @@ fn switchExpr( | |||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             const case_slice = case_scope.instructionsSlice(); |             const case_slice = case_scope.instructionsSlice(); | ||||||
|             const body_len = astgen.countBodyLenAfterFixups(case_slice); |             // Since we use the switch_block instruction itself to refer to the | ||||||
|  |             // capture, which will not be added to the child block, we need to | ||||||
|  |             // handle ref_table manually, and the same for the inline tag | ||||||
|  |             // capture instruction. | ||||||
|  |             const refs_len = refs: { | ||||||
|  |                 var n: usize = 0; | ||||||
|  |                 var check_inst = switch_block; | ||||||
|  |                 while (astgen.ref_table.get(check_inst)) |ref_inst| { | ||||||
|  |                     n += 1; | ||||||
|  |                     check_inst = ref_inst; | ||||||
|  |                 } | ||||||
|  |                 if (has_tag_capture) { | ||||||
|  |                     check_inst = tag_inst; | ||||||
|  |                     while (astgen.ref_table.get(check_inst)) |ref_inst| { | ||||||
|  |                         n += 1; | ||||||
|  |                         check_inst = ref_inst; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 break :refs n; | ||||||
|  |             }; | ||||||
|  |             const body_len = refs_len + astgen.countBodyLenAfterFixups(case_slice); | ||||||
|             try payloads.ensureUnusedCapacity(gpa, body_len); |             try payloads.ensureUnusedCapacity(gpa, body_len); | ||||||
|             const inline_bit = @as(u32, @intFromBool(case.inline_token != null)) << 31; |             payloads.items[body_len_index] = @bitCast(u32, Zir.Inst.SwitchBlock.ProngInfo{ | ||||||
|             payloads.items[body_len_index] = body_len | inline_bit; |                 .body_len = @intCast(u28, body_len), | ||||||
|  |                 .capture = capture, | ||||||
|  |                 .is_inline = case.inline_token != null, | ||||||
|  |                 .has_tag_capture = has_tag_capture, | ||||||
|  |             }); | ||||||
|  |             if (astgen.ref_table.fetchRemove(switch_block)) |kv| { | ||||||
|  |                 appendPossiblyRefdBodyInst(astgen, payloads, kv.value); | ||||||
|  |             } | ||||||
|  |             if (has_tag_capture) { | ||||||
|  |                 if (astgen.ref_table.fetchRemove(tag_inst)) |kv| { | ||||||
|  |                     appendPossiblyRefdBodyInst(astgen, payloads, kv.value); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|             appendBodyWithFixupsArrayList(astgen, payloads, case_slice); |             appendBodyWithFixupsArrayList(astgen, payloads, case_slice); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -7005,14 +7077,16 @@ fn switchExpr( | |||||||
| 
 | 
 | ||||||
|     try astgen.extra.ensureUnusedCapacity(gpa, @typeInfo(Zir.Inst.SwitchBlock).Struct.fields.len + |     try astgen.extra.ensureUnusedCapacity(gpa, @typeInfo(Zir.Inst.SwitchBlock).Struct.fields.len + | ||||||
|         @intFromBool(multi_cases_len != 0) + |         @intFromBool(multi_cases_len != 0) + | ||||||
|  |         @intFromBool(any_has_tag_capture) + | ||||||
|         payloads.items.len - case_table_end); |         payloads.items.len - case_table_end); | ||||||
| 
 | 
 | ||||||
|     const payload_index = astgen.addExtraAssumeCapacity(Zir.Inst.SwitchBlock{ |     const payload_index = astgen.addExtraAssumeCapacity(Zir.Inst.SwitchBlock{ | ||||||
|         .operand = cond, |         .operand = raw_operand, | ||||||
|         .bits = Zir.Inst.SwitchBlock.Bits{ |         .bits = Zir.Inst.SwitchBlock.Bits{ | ||||||
|             .has_multi_cases = multi_cases_len != 0, |             .has_multi_cases = multi_cases_len != 0, | ||||||
|             .has_else = special_prong == .@"else", |             .has_else = special_prong == .@"else", | ||||||
|             .has_under = special_prong == .under, |             .has_under = special_prong == .under, | ||||||
|  |             .any_has_tag_capture = any_has_tag_capture, | ||||||
|             .scalar_cases_len = @intCast(Zir.Inst.SwitchBlock.Bits.ScalarCasesLen, scalar_cases_len), |             .scalar_cases_len = @intCast(Zir.Inst.SwitchBlock.Bits.ScalarCasesLen, scalar_cases_len), | ||||||
|         }, |         }, | ||||||
|     }); |     }); | ||||||
| @ -7021,6 +7095,10 @@ fn switchExpr( | |||||||
|         astgen.extra.appendAssumeCapacity(multi_cases_len); |         astgen.extra.appendAssumeCapacity(multi_cases_len); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     if (any_has_tag_capture) { | ||||||
|  |         astgen.extra.appendAssumeCapacity(tag_inst); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     const zir_datas = astgen.instructions.items(.data); |     const zir_datas = astgen.instructions.items(.data); | ||||||
|     const zir_tags = astgen.instructions.items(.tag); |     const zir_tags = astgen.instructions.items(.tag); | ||||||
| 
 | 
 | ||||||
| @ -7043,7 +7121,7 @@ fn switchExpr( | |||||||
|             end_index += 3 + items_len + 2 * ranges_len; |             end_index += 3 + items_len + 2 * ranges_len; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         const body_len = @truncate(u31, payloads.items[body_len_index]); |         const body_len = @bitCast(Zir.Inst.SwitchBlock.ProngInfo, payloads.items[body_len_index]).body_len; | ||||||
|         end_index += body_len; |         end_index += body_len; | ||||||
| 
 | 
 | ||||||
|         switch (strat.tag) { |         switch (strat.tag) { | ||||||
| @ -7818,7 +7896,7 @@ fn unionInit( | |||||||
|     params: []const Ast.Node.Index, |     params: []const Ast.Node.Index, | ||||||
| ) InnerError!Zir.Inst.Ref { | ) InnerError!Zir.Inst.Ref { | ||||||
|     const union_type = try typeExpr(gz, scope, params[0]); |     const union_type = try typeExpr(gz, scope, params[0]); | ||||||
|     const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[1]); |     const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[1]); | ||||||
|     const field_type = try gz.addPlNode(.field_type_ref, params[1], Zir.Inst.FieldTypeRef{ |     const field_type = try gz.addPlNode(.field_type_ref, params[1], Zir.Inst.FieldTypeRef{ | ||||||
|         .container_type = union_type, |         .container_type = union_type, | ||||||
|         .field_name = field_name, |         .field_name = field_name, | ||||||
| @ -8040,12 +8118,12 @@ fn builtinCall( | |||||||
|             if (ri.rl == .ref) { |             if (ri.rl == .ref) { | ||||||
|                 return gz.addPlNode(.field_ptr_named, node, Zir.Inst.FieldNamed{ |                 return gz.addPlNode(.field_ptr_named, node, Zir.Inst.FieldNamed{ | ||||||
|                     .lhs = try expr(gz, scope, .{ .rl = .ref }, params[0]), |                     .lhs = try expr(gz, scope, .{ .rl = .ref }, params[0]), | ||||||
|                     .field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[1]), |                     .field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[1]), | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
|             const result = try gz.addPlNode(.field_val_named, node, Zir.Inst.FieldNamed{ |             const result = try gz.addPlNode(.field_val_named, node, Zir.Inst.FieldNamed{ | ||||||
|                 .lhs = try expr(gz, scope, .{ .rl = .none }, params[0]), |                 .lhs = try expr(gz, scope, .{ .rl = .none }, params[0]), | ||||||
|                 .field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[1]), |                 .field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[1]), | ||||||
|             }); |             }); | ||||||
|             return rvalue(gz, ri, result, node); |             return rvalue(gz, ri, result, node); | ||||||
|         }, |         }, | ||||||
| @ -8210,12 +8288,12 @@ fn builtinCall( | |||||||
|         .bit_size_of => return simpleUnOpType(gz, scope, ri, node, params[0], .bit_size_of), |         .bit_size_of => return simpleUnOpType(gz, scope, ri, node, params[0], .bit_size_of), | ||||||
|         .align_of    => return simpleUnOpType(gz, scope, ri, node, params[0], .align_of), |         .align_of    => return simpleUnOpType(gz, scope, ri, node, params[0], .align_of), | ||||||
| 
 | 
 | ||||||
|         .ptr_to_int            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .ptr_to_int), |         .int_from_ptr            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .int_from_ptr), | ||||||
|         .compile_error         => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[0], .compile_error), |         .compile_error         => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[0], .compile_error), | ||||||
|         .set_eval_branch_quota => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .coerced_ty = .u32_type } },    params[0], .set_eval_branch_quota), |         .set_eval_branch_quota => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .coerced_ty = .u32_type } },    params[0], .set_eval_branch_quota), | ||||||
|         .enum_to_int           => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .enum_to_int), |         .int_from_enum           => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .int_from_enum), | ||||||
|         .bool_to_int           => return simpleUnOp(gz, scope, ri, node, bool_ri,                                    params[0], .bool_to_int), |         .int_from_bool           => return simpleUnOp(gz, scope, ri, node, bool_ri,                                    params[0], .int_from_bool), | ||||||
|         .embed_file            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[0], .embed_file), |         .embed_file            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[0], .embed_file), | ||||||
|         .error_name            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .anyerror_type } },       params[0], .error_name), |         .error_name            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .anyerror_type } },       params[0], .error_name), | ||||||
|         .set_runtime_safety    => return simpleUnOp(gz, scope, ri, node, bool_ri,                                    params[0], .set_runtime_safety), |         .set_runtime_safety    => return simpleUnOp(gz, scope, ri, node, bool_ri,                                    params[0], .set_runtime_safety), | ||||||
|         .sqrt                  => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .sqrt), |         .sqrt                  => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .sqrt), | ||||||
| @ -8237,10 +8315,10 @@ fn builtinCall( | |||||||
|         .Frame                 => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .frame_type), |         .Frame                 => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .frame_type), | ||||||
|         .frame_size            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .frame_size), |         .frame_size            => return simpleUnOp(gz, scope, ri, node, .{ .rl = .none },                           params[0], .frame_size), | ||||||
| 
 | 
 | ||||||
|         .float_to_int => return typeCast(gz, scope, ri, node, params[0], params[1], .float_to_int), |         .int_from_float => return typeCast(gz, scope, ri, node, params[0], params[1], .int_from_float), | ||||||
|         .int_to_float => return typeCast(gz, scope, ri, node, params[0], params[1], .int_to_float), |         .float_from_int => return typeCast(gz, scope, ri, node, params[0], params[1], .float_from_int), | ||||||
|         .int_to_ptr   => return typeCast(gz, scope, ri, node, params[0], params[1], .int_to_ptr), |         .ptr_from_int   => return typeCast(gz, scope, ri, node, params[0], params[1], .ptr_from_int), | ||||||
|         .int_to_enum  => return typeCast(gz, scope, ri, node, params[0], params[1], .int_to_enum), |         .enum_from_int  => return typeCast(gz, scope, ri, node, params[0], params[1], .enum_from_int), | ||||||
|         .float_cast   => return typeCast(gz, scope, ri, node, params[0], params[1], .float_cast), |         .float_cast   => return typeCast(gz, scope, ri, node, params[0], params[1], .float_cast), | ||||||
|         .int_cast     => return typeCast(gz, scope, ri, node, params[0], params[1], .int_cast), |         .int_cast     => return typeCast(gz, scope, ri, node, params[0], params[1], .int_cast), | ||||||
|         .ptr_cast     => return typeCast(gz, scope, ri, node, params[0], params[1], .ptr_cast), |         .ptr_cast     => return typeCast(gz, scope, ri, node, params[0], params[1], .ptr_cast), | ||||||
| @ -8274,24 +8352,24 @@ fn builtinCall( | |||||||
|         }, |         }, | ||||||
|         .panic => { |         .panic => { | ||||||
|             try emitDbgNode(gz, node); |             try emitDbgNode(gz, node); | ||||||
|             return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[0], .panic); |             return simpleUnOp(gz, scope, ri, node, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[0], .panic); | ||||||
|         }, |         }, | ||||||
|         .trap => { |         .trap => { | ||||||
|             try emitDbgNode(gz, node); |             try emitDbgNode(gz, node); | ||||||
|             _ = try gz.addNode(.trap, node); |             _ = try gz.addNode(.trap, node); | ||||||
|             return rvalue(gz, ri, .void_value, node); |             return rvalue(gz, ri, .unreachable_value, node); | ||||||
|         }, |         }, | ||||||
|         .error_to_int => { |         .int_from_error => { | ||||||
|             const operand = try expr(gz, scope, .{ .rl = .none }, params[0]); |             const operand = try expr(gz, scope, .{ .rl = .none }, params[0]); | ||||||
|             const result = try gz.addExtendedPayload(.error_to_int, Zir.Inst.UnNode{ |             const result = try gz.addExtendedPayload(.int_from_error, Zir.Inst.UnNode{ | ||||||
|                 .node = gz.nodeIndexToRelative(node), |                 .node = gz.nodeIndexToRelative(node), | ||||||
|                 .operand = operand, |                 .operand = operand, | ||||||
|             }); |             }); | ||||||
|             return rvalue(gz, ri, result, node); |             return rvalue(gz, ri, result, node); | ||||||
|         }, |         }, | ||||||
|         .int_to_error => { |         .error_from_int => { | ||||||
|             const operand = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .u16_type } }, params[0]); |             const operand = try expr(gz, scope, .{ .rl = .{ .coerced_ty = .u16_type } }, params[0]); | ||||||
|             const result = try gz.addExtendedPayload(.int_to_error, Zir.Inst.UnNode{ |             const result = try gz.addExtendedPayload(.error_from_int, Zir.Inst.UnNode{ | ||||||
|                 .node = gz.nodeIndexToRelative(node), |                 .node = gz.nodeIndexToRelative(node), | ||||||
|                 .operand = operand, |                 .operand = operand, | ||||||
|             }); |             }); | ||||||
| @ -8390,7 +8468,7 @@ fn builtinCall( | |||||||
|         }, |         }, | ||||||
|         .c_define => { |         .c_define => { | ||||||
|             if (!gz.c_import) return gz.astgen.failNode(node, "C define valid only inside C import block", .{}); |             if (!gz.c_import) return gz.astgen.failNode(node, "C define valid only inside C import block", .{}); | ||||||
|             const name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[0]); |             const name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[0]); | ||||||
|             const value = try comptimeExpr(gz, scope, .{ .rl = .none }, params[1]); |             const value = try comptimeExpr(gz, scope, .{ .rl = .none }, params[1]); | ||||||
|             const result = try gz.addExtendedPayload(.c_define, Zir.Inst.BinNode{ |             const result = try gz.addExtendedPayload(.c_define, Zir.Inst.BinNode{ | ||||||
|                 .node = gz.nodeIndexToRelative(node), |                 .node = gz.nodeIndexToRelative(node), | ||||||
| @ -8470,8 +8548,8 @@ fn builtinCall( | |||||||
|             return rvalue(gz, ri, result, node); |             return rvalue(gz, ri, result, node); | ||||||
|         }, |         }, | ||||||
|         .call => { |         .call => { | ||||||
|             const modifier = try comptimeExpr(gz, scope, .{ .rl = .{ .coerced_ty = .modifier_type } }, params[0]); |             const modifier = try comptimeExpr(gz, scope, .{ .rl = .{ .coerced_ty = .call_modifier_type } }, params[0]); | ||||||
|             const callee = try calleeExpr(gz, scope, params[1]); |             const callee = try expr(gz, scope, .{ .rl = .none }, params[1]); | ||||||
|             const args = try expr(gz, scope, .{ .rl = .none }, params[2]); |             const args = try expr(gz, scope, .{ .rl = .none }, params[2]); | ||||||
|             const result = try gz.addPlNode(.builtin_call, node, Zir.Inst.BuiltinCall{ |             const result = try gz.addPlNode(.builtin_call, node, Zir.Inst.BuiltinCall{ | ||||||
|                 .modifier = modifier, |                 .modifier = modifier, | ||||||
| @ -8486,7 +8564,7 @@ fn builtinCall( | |||||||
|         }, |         }, | ||||||
|         .field_parent_ptr => { |         .field_parent_ptr => { | ||||||
|             const parent_type = try typeExpr(gz, scope, params[0]); |             const parent_type = try typeExpr(gz, scope, params[0]); | ||||||
|             const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[1]); |             const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, params[1]); | ||||||
|             const result = try gz.addPlNode(.field_parent_ptr, node, Zir.Inst.FieldParentPtr{ |             const result = try gz.addPlNode(.field_parent_ptr, node, Zir.Inst.FieldParentPtr{ | ||||||
|                 .parent_type = parent_type, |                 .parent_type = parent_type, | ||||||
|                 .field_name = field_name, |                 .field_name = field_name, | ||||||
| @ -8641,7 +8719,7 @@ fn hasDeclOrField( | |||||||
|     tag: Zir.Inst.Tag, |     tag: Zir.Inst.Tag, | ||||||
| ) InnerError!Zir.Inst.Ref { | ) InnerError!Zir.Inst.Ref { | ||||||
|     const container_type = try typeExpr(gz, scope, lhs_node); |     const container_type = try typeExpr(gz, scope, lhs_node); | ||||||
|     const name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, rhs_node); |     const name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, rhs_node); | ||||||
|     const result = try gz.addPlNode(tag, node, Zir.Inst.Bin{ |     const result = try gz.addPlNode(tag, node, Zir.Inst.Bin{ | ||||||
|         .lhs = container_type, |         .lhs = container_type, | ||||||
|         .rhs = name, |         .rhs = name, | ||||||
| @ -8698,7 +8776,7 @@ fn simpleUnOp( | |||||||
|     else |     else | ||||||
|         try expr(gz, scope, operand_ri, operand_node); |         try expr(gz, scope, operand_ri, operand_node); | ||||||
|     switch (tag) { |     switch (tag) { | ||||||
|         .tag_name, .error_name, .ptr_to_int => try emitDbgStmt(gz, cursor), |         .tag_name, .error_name, .int_from_ptr => try emitDbgStmt(gz, cursor), | ||||||
|         else => {}, |         else => {}, | ||||||
|     } |     } | ||||||
|     const result = try gz.addUnNode(tag, operand, node); |     const result = try gz.addUnNode(tag, operand, node); | ||||||
| @ -8791,7 +8869,7 @@ fn simpleCBuiltin( | |||||||
| ) InnerError!Zir.Inst.Ref { | ) InnerError!Zir.Inst.Ref { | ||||||
|     const name: []const u8 = if (tag == .c_undef) "C undef" else "C include"; |     const name: []const u8 = if (tag == .c_undef) "C undef" else "C include"; | ||||||
|     if (!gz.c_import) return gz.astgen.failNode(node, "{s} valid only inside C import block", .{name}); |     if (!gz.c_import) return gz.astgen.failNode(node, "{s} valid only inside C import block", .{name}); | ||||||
|     const operand = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, operand_node); |     const operand = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, operand_node); | ||||||
|     _ = try gz.addExtendedPayload(tag, Zir.Inst.UnNode{ |     _ = try gz.addExtendedPayload(tag, Zir.Inst.UnNode{ | ||||||
|         .node = gz.nodeIndexToRelative(node), |         .node = gz.nodeIndexToRelative(node), | ||||||
|         .operand = operand, |         .operand = operand, | ||||||
| @ -8809,7 +8887,7 @@ fn offsetOf( | |||||||
|     tag: Zir.Inst.Tag, |     tag: Zir.Inst.Tag, | ||||||
| ) InnerError!Zir.Inst.Ref { | ) InnerError!Zir.Inst.Ref { | ||||||
|     const type_inst = try typeExpr(gz, scope, lhs_node); |     const type_inst = try typeExpr(gz, scope, lhs_node); | ||||||
|     const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, rhs_node); |     const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .slice_const_u8_type } }, rhs_node); | ||||||
|     const result = try gz.addPlNode(tag, node, Zir.Inst.Bin{ |     const result = try gz.addPlNode(tag, node, Zir.Inst.Bin{ | ||||||
|         .lhs = type_inst, |         .lhs = type_inst, | ||||||
|         .rhs = field_name, |         .rhs = field_name, | ||||||
| @ -8931,7 +9009,10 @@ fn callExpr( | |||||||
|         } }); |         } }); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     assert(callee != .none); |     switch (callee) { | ||||||
|  |         .direct => |obj| assert(obj != .none), | ||||||
|  |         .field => |field| assert(field.obj_ptr != .none), | ||||||
|  |     } | ||||||
|     assert(node != 0); |     assert(node != 0); | ||||||
| 
 | 
 | ||||||
|     const call_index = @intCast(Zir.Inst.Index, astgen.instructions.len); |     const call_index = @intCast(Zir.Inst.Index, astgen.instructions.len); | ||||||
| @ -8970,89 +9051,98 @@ fn callExpr( | |||||||
|         else => false, |         else => false, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     const payload_index = try addExtra(astgen, Zir.Inst.Call{ |     switch (callee) { | ||||||
|         .callee = callee, |         .direct => |callee_obj| { | ||||||
|         .flags = .{ |             const payload_index = try addExtra(astgen, Zir.Inst.Call{ | ||||||
|             .pop_error_return_trace = !propagate_error_trace, |                 .callee = callee_obj, | ||||||
|             .packed_modifier = @intCast(Zir.Inst.Call.Flags.PackedModifier, @intFromEnum(modifier)), |                 .flags = .{ | ||||||
|             .args_len = @intCast(Zir.Inst.Call.Flags.PackedArgsLen, call.ast.params.len), |                     .pop_error_return_trace = !propagate_error_trace, | ||||||
|  |                     .packed_modifier = @intCast(Zir.Inst.Call.Flags.PackedModifier, @intFromEnum(modifier)), | ||||||
|  |                     .args_len = @intCast(Zir.Inst.Call.Flags.PackedArgsLen, call.ast.params.len), | ||||||
|  |                 }, | ||||||
|  |             }); | ||||||
|  |             if (call.ast.params.len != 0) { | ||||||
|  |                 try astgen.extra.appendSlice(astgen.gpa, astgen.scratch.items[scratch_top..]); | ||||||
|  |             } | ||||||
|  |             gz.astgen.instructions.set(call_index, .{ | ||||||
|  |                 .tag = .call, | ||||||
|  |                 .data = .{ .pl_node = .{ | ||||||
|  |                     .src_node = gz.nodeIndexToRelative(node), | ||||||
|  |                     .payload_index = payload_index, | ||||||
|  |                 } }, | ||||||
|  |             }); | ||||||
|  |         }, | ||||||
|  |         .field => |callee_field| { | ||||||
|  |             const payload_index = try addExtra(astgen, Zir.Inst.FieldCall{ | ||||||
|  |                 .obj_ptr = callee_field.obj_ptr, | ||||||
|  |                 .field_name_start = callee_field.field_name_start, | ||||||
|  |                 .flags = .{ | ||||||
|  |                     .pop_error_return_trace = !propagate_error_trace, | ||||||
|  |                     .packed_modifier = @intCast(Zir.Inst.Call.Flags.PackedModifier, @intFromEnum(modifier)), | ||||||
|  |                     .args_len = @intCast(Zir.Inst.Call.Flags.PackedArgsLen, call.ast.params.len), | ||||||
|  |                 }, | ||||||
|  |             }); | ||||||
|  |             if (call.ast.params.len != 0) { | ||||||
|  |                 try astgen.extra.appendSlice(astgen.gpa, astgen.scratch.items[scratch_top..]); | ||||||
|  |             } | ||||||
|  |             gz.astgen.instructions.set(call_index, .{ | ||||||
|  |                 .tag = .field_call, | ||||||
|  |                 .data = .{ .pl_node = .{ | ||||||
|  |                     .src_node = gz.nodeIndexToRelative(node), | ||||||
|  |                     .payload_index = payload_index, | ||||||
|  |                 } }, | ||||||
|  |             }); | ||||||
|         }, |         }, | ||||||
|     }); |  | ||||||
|     if (call.ast.params.len != 0) { |  | ||||||
|         try astgen.extra.appendSlice(astgen.gpa, astgen.scratch.items[scratch_top..]); |  | ||||||
|     } |     } | ||||||
|     gz.astgen.instructions.set(call_index, .{ |  | ||||||
|         .tag = .call, |  | ||||||
|         .data = .{ .pl_node = .{ |  | ||||||
|             .src_node = gz.nodeIndexToRelative(node), |  | ||||||
|             .payload_index = payload_index, |  | ||||||
|         } }, |  | ||||||
|     }); |  | ||||||
|     return rvalue(gz, ri, call_inst, node); // TODO function call with result location |     return rvalue(gz, ri, call_inst, node); // TODO function call with result location | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// calleeExpr generates the function part of a call expression (f in f(x)), or the | const Callee = union(enum) { | ||||||
| /// callee argument to the @call() builtin. If the lhs is a field access or the |     field: struct { | ||||||
| /// @field() builtin, we need to generate a special field_call_bind instruction |         /// A *pointer* to the object the field is fetched on, so that we can | ||||||
| /// instead of the normal field_val or field_ptr.  If this is a inst.func() call, |         /// promote the lvalue to an address if the first parameter requires it. | ||||||
| /// this instruction will capture the value of the first argument before evaluating |         obj_ptr: Zir.Inst.Ref, | ||||||
| /// the other arguments. We need to use .ref here to guarantee we will be able to |         /// Offset into `string_bytes`. | ||||||
| /// promote an lvalue to an address if the first parameter requires it.  This |         field_name_start: u32, | ||||||
| /// unfortunately also means we need to take a reference to any types on the lhs. |     }, | ||||||
|  |     direct: Zir.Inst.Ref, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /// calleeExpr generates the function part of a call expression (f in f(x)), but | ||||||
|  | /// *not* the callee argument to the @call() builtin. Its purpose is to | ||||||
|  | /// distinguish between standard calls and method call syntax `a.b()`. Thus, if | ||||||
|  | /// the lhs is a field access, we return using the `field` union field; | ||||||
|  | /// otherwise, we use the `direct` union field. | ||||||
| fn calleeExpr( | fn calleeExpr( | ||||||
|     gz: *GenZir, |     gz: *GenZir, | ||||||
|     scope: *Scope, |     scope: *Scope, | ||||||
|     node: Ast.Node.Index, |     node: Ast.Node.Index, | ||||||
| ) InnerError!Zir.Inst.Ref { | ) InnerError!Callee { | ||||||
|     const astgen = gz.astgen; |     const astgen = gz.astgen; | ||||||
|     const tree = astgen.tree; |     const tree = astgen.tree; | ||||||
| 
 | 
 | ||||||
|     const tag = tree.nodes.items(.tag)[node]; |     const tag = tree.nodes.items(.tag)[node]; | ||||||
|     switch (tag) { |     switch (tag) { | ||||||
|         .field_access => return addFieldAccess(.field_call_bind, gz, scope, .{ .rl = .ref }, node), |         .field_access => { | ||||||
| 
 |  | ||||||
|         .builtin_call_two, |  | ||||||
|         .builtin_call_two_comma, |  | ||||||
|         .builtin_call, |  | ||||||
|         .builtin_call_comma, |  | ||||||
|         => { |  | ||||||
|             const node_datas = tree.nodes.items(.data); |  | ||||||
|             const main_tokens = tree.nodes.items(.main_token); |             const main_tokens = tree.nodes.items(.main_token); | ||||||
|             const builtin_token = main_tokens[node]; |             const node_datas = tree.nodes.items(.data); | ||||||
|             const builtin_name = tree.tokenSlice(builtin_token); |             const object_node = node_datas[node].lhs; | ||||||
|  |             const dot_token = main_tokens[node]; | ||||||
|  |             const field_ident = dot_token + 1; | ||||||
|  |             const str_index = try astgen.identAsString(field_ident); | ||||||
|  |             // Capture the object by reference so we can promote it to an | ||||||
|  |             // address in Sema if needed. | ||||||
|  |             const lhs = try expr(gz, scope, .{ .rl = .ref }, object_node); | ||||||
| 
 | 
 | ||||||
|             var inline_params: [2]Ast.Node.Index = undefined; |             const cursor = maybeAdvanceSourceCursorToMainToken(gz, node); | ||||||
|             var params: []Ast.Node.Index = switch (tag) { |             try emitDbgStmt(gz, cursor); | ||||||
|                 .builtin_call, |  | ||||||
|                 .builtin_call_comma, |  | ||||||
|                 => tree.extra_data[node_datas[node].lhs..node_datas[node].rhs], |  | ||||||
| 
 | 
 | ||||||
|                 .builtin_call_two, |             return .{ .field = .{ | ||||||
|                 .builtin_call_two_comma, |                 .obj_ptr = lhs, | ||||||
|                 => blk: { |                 .field_name_start = str_index, | ||||||
|                     inline_params = .{ node_datas[node].lhs, node_datas[node].rhs }; |             } }; | ||||||
|                     const len: usize = if (inline_params[0] == 0) @as(usize, 0) else if (inline_params[1] == 0) @as(usize, 1) else @as(usize, 2); |  | ||||||
|                     break :blk inline_params[0..len]; |  | ||||||
|                 }, |  | ||||||
| 
 |  | ||||||
|                 else => unreachable, |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             // If anything is wrong, fall back to builtinCall. |  | ||||||
|             // It will emit any necessary compile errors and notes. |  | ||||||
|             if (std.mem.eql(u8, builtin_name, "@field") and params.len == 2) { |  | ||||||
|                 const lhs = try expr(gz, scope, .{ .rl = .ref }, params[0]); |  | ||||||
|                 const field_name = try comptimeExpr(gz, scope, .{ .rl = .{ .ty = .const_slice_u8_type } }, params[1]); |  | ||||||
|                 return gz.addExtendedPayload(.field_call_bind_named, Zir.Inst.FieldNamedNode{ |  | ||||||
|                     .node = gz.nodeIndexToRelative(node), |  | ||||||
|                     .lhs = lhs, |  | ||||||
|                     .field_name = field_name, |  | ||||||
|                 }); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return builtinCall(gz, scope, .{ .rl = .none }, node, params); |  | ||||||
|         }, |         }, | ||||||
|         else => return expr(gz, scope, .{ .rl = .none }, node), |         else => return .{ .direct = try expr(gz, scope, .{ .rl = .none }, node) }, | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -10199,6 +10289,8 @@ fn rvalue( | |||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.i32_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.i32_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.u64_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.u64_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.i64_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.i64_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.u128_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.i128_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.usize_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.usize_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.isize_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.isize_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.c_char_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.c_char_type), | ||||||
| @ -10224,15 +10316,30 @@ fn rvalue( | |||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.comptime_int_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.comptime_int_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.comptime_float_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.comptime_float_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.noreturn_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.noreturn_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.anyframe_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.null_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.null_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.undefined_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.undefined_type), | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.fn_noreturn_no_args_type), |  | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.fn_void_no_args_type), |  | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.fn_naked_noreturn_no_args_type), |  | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.fn_ccc_void_no_args_type), |  | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.single_const_pointer_to_comptime_int_type), |  | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.const_slice_u8_type), |  | ||||||
|                 as_ty | @intFromEnum(Zir.Inst.Ref.enum_literal_type), |                 as_ty | @intFromEnum(Zir.Inst.Ref.enum_literal_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.atomic_order_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.atomic_rmw_op_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.calling_convention_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.address_space_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.float_mode_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.reduce_op_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.call_modifier_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.prefetch_options_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.export_options_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.extern_options_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.type_info_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.manyptr_u8_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.manyptr_const_u8_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.manyptr_const_u8_sentinel_0_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.single_const_pointer_to_comptime_int_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.slice_const_u8_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.slice_const_u8_sentinel_0_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.anyerror_void_error_union_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.generic_poison_type), | ||||||
|  |                 as_ty | @intFromEnum(Zir.Inst.Ref.empty_struct_type), | ||||||
|                 as_comptime_int | @intFromEnum(Zir.Inst.Ref.zero), |                 as_comptime_int | @intFromEnum(Zir.Inst.Ref.zero), | ||||||
|                 as_comptime_int | @intFromEnum(Zir.Inst.Ref.one), |                 as_comptime_int | @intFromEnum(Zir.Inst.Ref.one), | ||||||
|                 as_bool | @intFromEnum(Zir.Inst.Ref.bool_true), |                 as_bool | @intFromEnum(Zir.Inst.Ref.bool_true), | ||||||
| @ -10605,8 +10712,8 @@ fn identAsString(astgen: *AstGen, ident_token: Ast.TokenIndex) !u32 { | |||||||
|     const string_bytes = &astgen.string_bytes; |     const string_bytes = &astgen.string_bytes; | ||||||
|     const str_index = @intCast(u32, string_bytes.items.len); |     const str_index = @intCast(u32, string_bytes.items.len); | ||||||
|     try astgen.appendIdentStr(ident_token, string_bytes); |     try astgen.appendIdentStr(ident_token, string_bytes); | ||||||
|     const key = string_bytes.items[str_index..]; |     const key: []const u8 = string_bytes.items[str_index..]; | ||||||
|     const gop = try astgen.string_table.getOrPutContextAdapted(gpa, @as([]const u8, key), StringIndexAdapter{ |     const gop = try astgen.string_table.getOrPutContextAdapted(gpa, key, StringIndexAdapter{ | ||||||
|         .bytes = string_bytes, |         .bytes = string_bytes, | ||||||
|     }, StringIndexContext{ |     }, StringIndexContext{ | ||||||
|         .bytes = string_bytes, |         .bytes = string_bytes, | ||||||
| @ -11689,9 +11796,9 @@ const GenZir = struct { | |||||||
|     ) !Zir.Inst.Index { |     ) !Zir.Inst.Index { | ||||||
|         const gpa = gz.astgen.gpa; |         const gpa = gz.astgen.gpa; | ||||||
|         const param_body = param_gz.instructionsSlice(); |         const param_body = param_gz.instructionsSlice(); | ||||||
|  |         const body_len = gz.astgen.countBodyLenAfterFixups(param_body); | ||||||
|         try gz.astgen.instructions.ensureUnusedCapacity(gpa, 1); |         try gz.astgen.instructions.ensureUnusedCapacity(gpa, 1); | ||||||
|         try gz.astgen.extra.ensureUnusedCapacity(gpa, @typeInfo(Zir.Inst.Param).Struct.fields.len + |         try gz.astgen.extra.ensureUnusedCapacity(gpa, @typeInfo(Zir.Inst.Param).Struct.fields.len + body_len); | ||||||
|             param_body.len); |  | ||||||
| 
 | 
 | ||||||
|         const doc_comment_index = if (first_doc_comment) |first| |         const doc_comment_index = if (first_doc_comment) |first| | ||||||
|             try gz.astgen.docCommentAsStringFromFirst(abs_tok_index, first) |             try gz.astgen.docCommentAsStringFromFirst(abs_tok_index, first) | ||||||
| @ -11701,9 +11808,9 @@ const GenZir = struct { | |||||||
|         const payload_index = gz.astgen.addExtraAssumeCapacity(Zir.Inst.Param{ |         const payload_index = gz.astgen.addExtraAssumeCapacity(Zir.Inst.Param{ | ||||||
|             .name = name, |             .name = name, | ||||||
|             .doc_comment = doc_comment_index, |             .doc_comment = doc_comment_index, | ||||||
|             .body_len = @intCast(u32, param_body.len), |             .body_len = @intCast(u32, body_len), | ||||||
|         }); |         }); | ||||||
|         gz.astgen.extra.appendSliceAssumeCapacity(param_body); |         gz.astgen.appendBodyWithFixups(param_body); | ||||||
|         param_gz.unstack(); |         param_gz.unstack(); | ||||||
| 
 | 
 | ||||||
|         const new_index = @intCast(Zir.Inst.Index, gz.astgen.instructions.len); |         const new_index = @intCast(Zir.Inst.Index, gz.astgen.instructions.len); | ||||||
|  | |||||||
| @ -12,7 +12,7 @@ pub const Tag = enum { | |||||||
|     atomic_store, |     atomic_store, | ||||||
|     bit_cast, |     bit_cast, | ||||||
|     bit_offset_of, |     bit_offset_of, | ||||||
|     bool_to_int, |     int_from_bool, | ||||||
|     bit_size_of, |     bit_size_of, | ||||||
|     breakpoint, |     breakpoint, | ||||||
|     mul_add, |     mul_add, | ||||||
| @ -39,10 +39,10 @@ pub const Tag = enum { | |||||||
|     div_floor, |     div_floor, | ||||||
|     div_trunc, |     div_trunc, | ||||||
|     embed_file, |     embed_file, | ||||||
|     enum_to_int, |     int_from_enum, | ||||||
|     error_name, |     error_name, | ||||||
|     error_return_trace, |     error_return_trace, | ||||||
|     error_to_int, |     int_from_error, | ||||||
|     err_set_cast, |     err_set_cast, | ||||||
|     @"export", |     @"export", | ||||||
|     @"extern", |     @"extern", | ||||||
| @ -50,7 +50,7 @@ pub const Tag = enum { | |||||||
|     field, |     field, | ||||||
|     field_parent_ptr, |     field_parent_ptr, | ||||||
|     float_cast, |     float_cast, | ||||||
|     float_to_int, |     int_from_float, | ||||||
|     frame, |     frame, | ||||||
|     Frame, |     Frame, | ||||||
|     frame_address, |     frame_address, | ||||||
| @ -60,10 +60,10 @@ pub const Tag = enum { | |||||||
|     import, |     import, | ||||||
|     in_comptime, |     in_comptime, | ||||||
|     int_cast, |     int_cast, | ||||||
|     int_to_enum, |     enum_from_int, | ||||||
|     int_to_error, |     error_from_int, | ||||||
|     int_to_float, |     float_from_int, | ||||||
|     int_to_ptr, |     ptr_from_int, | ||||||
|     max, |     max, | ||||||
|     memcpy, |     memcpy, | ||||||
|     memset, |     memset, | ||||||
| @ -76,7 +76,7 @@ pub const Tag = enum { | |||||||
|     pop_count, |     pop_count, | ||||||
|     prefetch, |     prefetch, | ||||||
|     ptr_cast, |     ptr_cast, | ||||||
|     ptr_to_int, |     int_from_ptr, | ||||||
|     rem, |     rem, | ||||||
|     return_address, |     return_address, | ||||||
|     select, |     select, | ||||||
| @ -238,9 +238,9 @@ pub const list = list: { | |||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@boolToInt", |             "@intFromBool", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .bool_to_int, |                 .tag = .int_from_bool, | ||||||
|                 .param_count = 1, |                 .param_count = 1, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
| @ -425,9 +425,9 @@ pub const list = list: { | |||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@enumToInt", |             "@intFromEnum", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .enum_to_int, |                 .tag = .int_from_enum, | ||||||
|                 .param_count = 1, |                 .param_count = 1, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
| @ -446,9 +446,9 @@ pub const list = list: { | |||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@errorToInt", |             "@intFromError", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .error_to_int, |                 .tag = .int_from_error, | ||||||
|                 .param_count = 1, |                 .param_count = 1, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
| @ -506,9 +506,9 @@ pub const list = list: { | |||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@floatToInt", |             "@intFromFloat", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .float_to_int, |                 .tag = .int_from_float, | ||||||
|                 .param_count = 2, |                 .param_count = 2, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
| @ -576,31 +576,31 @@ pub const list = list: { | |||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@intToEnum", |             "@enumFromInt", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .int_to_enum, |                 .tag = .enum_from_int, | ||||||
|                 .param_count = 2, |                 .param_count = 2, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@intToError", |             "@errorFromInt", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .int_to_error, |                 .tag = .error_from_int, | ||||||
|                 .eval_to_error = .always, |                 .eval_to_error = .always, | ||||||
|                 .param_count = 1, |                 .param_count = 1, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@intToFloat", |             "@floatFromInt", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .int_to_float, |                 .tag = .float_from_int, | ||||||
|                 .param_count = 2, |                 .param_count = 2, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@intToPtr", |             "@ptrFromInt", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .int_to_ptr, |                 .tag = .ptr_from_int, | ||||||
|                 .param_count = 2, |                 .param_count = 2, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
| @ -689,9 +689,9 @@ pub const list = list: { | |||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|         .{ |         .{ | ||||||
|             "@ptrToInt", |             "@intFromPtr", | ||||||
|             .{ |             .{ | ||||||
|                 .tag = .ptr_to_int, |                 .tag = .int_from_ptr, | ||||||
|                 .param_count = 1, |                 .param_count = 1, | ||||||
|             }, |             }, | ||||||
|         }, |         }, | ||||||
|  | |||||||
| @ -19,6 +19,7 @@ const BigIntConst = std.math.big.int.Const; | |||||||
| const BigIntMutable = std.math.big.int.Mutable; | const BigIntMutable = std.math.big.int.Mutable; | ||||||
| const Ast = std.zig.Ast; | const Ast = std.zig.Ast; | ||||||
| 
 | 
 | ||||||
|  | const InternPool = @import("../analyser/InternPool.zig"); | ||||||
| const Zir = @This(); | const Zir = @This(); | ||||||
| const Module = @import("Module.zig"); | const Module = @import("Module.zig"); | ||||||
| const LazySrcLoc = Module.LazySrcLoc; | const LazySrcLoc = Module.LazySrcLoc; | ||||||
| @ -294,6 +295,14 @@ pub const Inst = struct { | |||||||
|         /// Uses the `pl_node` union field with payload `Call`. |         /// Uses the `pl_node` union field with payload `Call`. | ||||||
|         /// AST node is the function call. |         /// AST node is the function call. | ||||||
|         call, |         call, | ||||||
|  |         /// Function call using `a.b()` syntax. | ||||||
|  |         /// Uses the named field as the callee. If there is no such field, searches in the type for | ||||||
|  |         /// a decl matching the field name. The decl is resolved and we ensure that it's a function | ||||||
|  |         /// which can accept the object as the first parameter, with one pointer fixup. This | ||||||
|  |         /// function is then used as the callee, with the object as an implicit first parameter. | ||||||
|  |         /// Uses the `pl_node` union field with payload `FieldCall`. | ||||||
|  |         /// AST node is the function call. | ||||||
|  |         field_call, | ||||||
|         /// Implements the `@call` builtin. |         /// Implements the `@call` builtin. | ||||||
|         /// Uses the `pl_node` union field with payload `BuiltinCall`. |         /// Uses the `pl_node` union field with payload `BuiltinCall`. | ||||||
|         /// AST node is the builtin call. |         /// AST node is the builtin call. | ||||||
| @ -429,15 +438,6 @@ pub const Inst = struct { | |||||||
|         /// This instruction also accepts a pointer. |         /// This instruction also accepts a pointer. | ||||||
|         /// Uses `pl_node` field. The AST node is the a.b syntax. Payload is Field. |         /// Uses `pl_node` field. The AST node is the a.b syntax. Payload is Field. | ||||||
|         field_val, |         field_val, | ||||||
|         /// Given a pointer to a struct or object that contains virtual fields, returns the |  | ||||||
|         /// named field.  If there is no named field, searches in the type for a decl that |  | ||||||
|         /// matches the field name.  The decl is resolved and we ensure that it's a function |  | ||||||
|         /// which can accept the object as the first parameter, with one pointer fixup.  If |  | ||||||
|         /// all of that works, this instruction produces a special "bound function" value |  | ||||||
|         /// which contains both the function and the saved first parameter value. |  | ||||||
|         /// Bound functions may only be used as the function parameter to a `call` or |  | ||||||
|         /// `builtin_call` instruction.  Any other use is invalid zir and may crash the compiler. |  | ||||||
|         field_call_bind, |  | ||||||
|         /// Given a pointer to a struct or object that contains virtual fields, returns a pointer |         /// Given a pointer to a struct or object that contains virtual fields, returns a pointer | ||||||
|         /// to the named field. The field name is a comptime instruction. Used by @field. |         /// to the named field. The field name is a comptime instruction. Used by @field. | ||||||
|         /// Uses `pl_node` field. The AST node is the builtin call. Payload is FieldNamed. |         /// Uses `pl_node` field. The AST node is the builtin call. Payload is FieldNamed. | ||||||
| @ -567,6 +567,10 @@ pub const Inst = struct { | |||||||
|         /// Returns a pointer to the subslice. |         /// Returns a pointer to the subslice. | ||||||
|         /// Uses the `pl_node` field. AST node is the slice syntax. Payload is `SliceSentinel`. |         /// Uses the `pl_node` field. AST node is the slice syntax. Payload is `SliceSentinel`. | ||||||
|         slice_sentinel, |         slice_sentinel, | ||||||
|  |         /// Slice operation `array_ptr[start..][0..len]`. Optional sentinel. | ||||||
|  |         /// Returns a pointer to the subslice. | ||||||
|  |         /// Uses the `pl_node` field. AST node is the slice syntax. Payload is `SliceLength`. | ||||||
|  |         slice_length, | ||||||
|         /// Write a value to a pointer. For loading, see `load`. |         /// Write a value to a pointer. For loading, see `load`. | ||||||
|         /// Source location is assumed to be same as previous instruction. |         /// Source location is assumed to be same as previous instruction. | ||||||
|         /// Uses the `bin` union field. |         /// Uses the `bin` union field. | ||||||
| @ -660,38 +664,9 @@ pub const Inst = struct { | |||||||
|         /// A switch expression. Uses the `pl_node` union field. |         /// A switch expression. Uses the `pl_node` union field. | ||||||
|         /// AST node is the switch, payload is `SwitchBlock`. |         /// AST node is the switch, payload is `SwitchBlock`. | ||||||
|         switch_block, |         switch_block, | ||||||
|         /// Produces the value that will be switched on. For example, for |         /// A switch expression. Uses the `pl_node` union field. | ||||||
|         /// integers, it returns the integer with no modifications. For tagged unions, it |         /// AST node is the switch, payload is `SwitchBlock`. Operand is a pointer. | ||||||
|         /// returns the active enum tag. |         switch_block_ref, | ||||||
|         /// Uses the `un_node` union field. |  | ||||||
|         switch_cond, |  | ||||||
|         /// Same as `switch_cond`, except the input operand is a pointer to |  | ||||||
|         /// what will be switched on. |  | ||||||
|         /// Uses the `un_node` union field. |  | ||||||
|         switch_cond_ref, |  | ||||||
|         /// Produces the capture value for a switch prong. |  | ||||||
|         /// Uses the `switch_capture` field. |  | ||||||
|         /// If the `prong_index` field is max int, it means this is the capture |  | ||||||
|         /// for the else/`_` prong. |  | ||||||
|         switch_capture, |  | ||||||
|         /// Produces the capture value for a switch prong. |  | ||||||
|         /// Result is a pointer to the value. |  | ||||||
|         /// Uses the `switch_capture` field. |  | ||||||
|         /// If the `prong_index` field is max int, it means this is the capture |  | ||||||
|         /// for the else/`_` prong. |  | ||||||
|         switch_capture_ref, |  | ||||||
|         /// Produces the capture value for a switch prong. |  | ||||||
|         /// The prong is one of the multi cases. |  | ||||||
|         /// Uses the `switch_capture` field. |  | ||||||
|         switch_capture_multi, |  | ||||||
|         /// Produces the capture value for a switch prong. |  | ||||||
|         /// The prong is one of the multi cases. |  | ||||||
|         /// Result is a pointer to the value. |  | ||||||
|         /// Uses the `switch_capture` field. |  | ||||||
|         switch_capture_multi_ref, |  | ||||||
|         /// Produces the capture value for an inline switch prong tag capture. |  | ||||||
|         /// Uses the `un_tok` field. |  | ||||||
|         switch_capture_tag, |  | ||||||
|         /// Given a |         /// Given a | ||||||
|         ///   *A returns *A |         ///   *A returns *A | ||||||
|         ///   *E!A returns *A |         ///   *E!A returns *A | ||||||
| @ -771,9 +746,9 @@ pub const Inst = struct { | |||||||
|         /// Implements the `@bitSizeOf` builtin. Uses `un_node`. |         /// Implements the `@bitSizeOf` builtin. Uses `un_node`. | ||||||
|         bit_size_of, |         bit_size_of, | ||||||
| 
 | 
 | ||||||
|         /// Implement builtin `@ptrToInt`. Uses `un_node`. |         /// Implement builtin `@intFromPtr`. Uses `un_node`. | ||||||
|         /// Convert a pointer to a `usize` integer. |         /// Convert a pointer to a `usize` integer. | ||||||
|         ptr_to_int, |         int_from_ptr, | ||||||
|         /// Emit an error message and fail compilation. |         /// Emit an error message and fail compilation. | ||||||
|         /// Uses the `un_node` field. |         /// Uses the `un_node` field. | ||||||
|         compile_error, |         compile_error, | ||||||
| @ -783,11 +758,11 @@ pub const Inst = struct { | |||||||
|         set_eval_branch_quota, |         set_eval_branch_quota, | ||||||
|         /// Converts an enum value into an integer. Resulting type will be the tag type |         /// Converts an enum value into an integer. Resulting type will be the tag type | ||||||
|         /// of the enum. Uses `un_node`. |         /// of the enum. Uses `un_node`. | ||||||
|         enum_to_int, |         int_from_enum, | ||||||
|         /// Implement builtin `@alignOf`. Uses `un_node`. |         /// Implement builtin `@alignOf`. Uses `un_node`. | ||||||
|         align_of, |         align_of, | ||||||
|         /// Implement builtin `@boolToInt`. Uses `un_node`. |         /// Implement builtin `@intFromBool`. Uses `un_node`. | ||||||
|         bool_to_int, |         int_from_bool, | ||||||
|         /// Implement builtin `@embedFile`. Uses `un_node`. |         /// Implement builtin `@embedFile`. Uses `un_node`. | ||||||
|         embed_file, |         embed_file, | ||||||
|         /// Implement builtin `@errorName`. Uses `un_node`. |         /// Implement builtin `@errorName`. Uses `un_node`. | ||||||
| @ -836,18 +811,18 @@ pub const Inst = struct { | |||||||
|         /// Implement builtin `@frameSize`. Uses `un_node`. |         /// Implement builtin `@frameSize`. Uses `un_node`. | ||||||
|         frame_size, |         frame_size, | ||||||
| 
 | 
 | ||||||
|         /// Implements the `@floatToInt` builtin. |         /// Implements the `@intFromFloat` builtin. | ||||||
|         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. |         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. | ||||||
|         float_to_int, |         int_from_float, | ||||||
|         /// Implements the `@intToFloat` builtin. |         /// Implements the `@floatFromInt` builtin. | ||||||
|         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. |         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. | ||||||
|         int_to_float, |         float_from_int, | ||||||
|         /// Implements the `@intToPtr` builtin. |         /// Implements the `@ptrFromInt` builtin. | ||||||
|         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. |         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. | ||||||
|         int_to_ptr, |         ptr_from_int, | ||||||
|         /// Converts an integer into an enum value. |         /// Converts an integer into an enum value. | ||||||
|         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. |         /// Uses `pl_node` with payload `Bin`. `lhs` is dest type, `rhs` is operand. | ||||||
|         int_to_enum, |         enum_from_int, | ||||||
|         /// Convert a larger float type to any other float type, possibly causing |         /// Convert a larger float type to any other float type, possibly causing | ||||||
|         /// a loss of precision. |         /// a loss of precision. | ||||||
|         /// Uses the `pl_node` field. AST is the `@floatCast` syntax. |         /// Uses the `pl_node` field. AST is the `@floatCast` syntax. | ||||||
| @ -1044,6 +1019,7 @@ pub const Inst = struct { | |||||||
|                 .bool_br_or, |                 .bool_br_or, | ||||||
|                 .bool_not, |                 .bool_not, | ||||||
|                 .call, |                 .call, | ||||||
|  |                 .field_call, | ||||||
|                 .cmp_lt, |                 .cmp_lt, | ||||||
|                 .cmp_lte, |                 .cmp_lte, | ||||||
|                 .cmp_eq, |                 .cmp_eq, | ||||||
| @ -1076,7 +1052,6 @@ pub const Inst = struct { | |||||||
|                 .field_ptr, |                 .field_ptr, | ||||||
|                 .field_ptr_init, |                 .field_ptr_init, | ||||||
|                 .field_val, |                 .field_val, | ||||||
|                 .field_call_bind, |  | ||||||
|                 .field_ptr_named, |                 .field_ptr_named, | ||||||
|                 .field_val_named, |                 .field_val_named, | ||||||
|                 .func, |                 .func, | ||||||
| @ -1132,18 +1107,13 @@ pub const Inst = struct { | |||||||
|                 .slice_start, |                 .slice_start, | ||||||
|                 .slice_end, |                 .slice_end, | ||||||
|                 .slice_sentinel, |                 .slice_sentinel, | ||||||
|  |                 .slice_length, | ||||||
|                 .import, |                 .import, | ||||||
|                 .typeof_log2_int_type, |                 .typeof_log2_int_type, | ||||||
|                 .resolve_inferred_alloc, |                 .resolve_inferred_alloc, | ||||||
|                 .set_eval_branch_quota, |                 .set_eval_branch_quota, | ||||||
|                 .switch_capture, |  | ||||||
|                 .switch_capture_ref, |  | ||||||
|                 .switch_capture_multi, |  | ||||||
|                 .switch_capture_multi_ref, |  | ||||||
|                 .switch_capture_tag, |  | ||||||
|                 .switch_block, |                 .switch_block, | ||||||
|                 .switch_cond, |                 .switch_block_ref, | ||||||
|                 .switch_cond_ref, |  | ||||||
|                 .array_base_ptr, |                 .array_base_ptr, | ||||||
|                 .field_base_ptr, |                 .field_base_ptr, | ||||||
|                 .validate_array_init_ty, |                 .validate_array_init_ty, | ||||||
| @ -1163,14 +1133,14 @@ pub const Inst = struct { | |||||||
|                 .union_init, |                 .union_init, | ||||||
|                 .field_type, |                 .field_type, | ||||||
|                 .field_type_ref, |                 .field_type_ref, | ||||||
|                 .int_to_enum, |                 .enum_from_int, | ||||||
|                 .enum_to_int, |                 .int_from_enum, | ||||||
|                 .type_info, |                 .type_info, | ||||||
|                 .size_of, |                 .size_of, | ||||||
|                 .bit_size_of, |                 .bit_size_of, | ||||||
|                 .ptr_to_int, |                 .int_from_ptr, | ||||||
|                 .align_of, |                 .align_of, | ||||||
|                 .bool_to_int, |                 .int_from_bool, | ||||||
|                 .embed_file, |                 .embed_file, | ||||||
|                 .error_name, |                 .error_name, | ||||||
|                 .set_runtime_safety, |                 .set_runtime_safety, | ||||||
| @ -1192,9 +1162,9 @@ pub const Inst = struct { | |||||||
|                 .type_name, |                 .type_name, | ||||||
|                 .frame_type, |                 .frame_type, | ||||||
|                 .frame_size, |                 .frame_size, | ||||||
|                 .float_to_int, |                 .int_from_float, | ||||||
|                 .int_to_float, |                 .float_from_int, | ||||||
|                 .int_to_ptr, |                 .ptr_from_int, | ||||||
|                 .float_cast, |                 .float_cast, | ||||||
|                 .int_cast, |                 .int_cast, | ||||||
|                 .ptr_cast, |                 .ptr_cast, | ||||||
| @ -1353,6 +1323,7 @@ pub const Inst = struct { | |||||||
|                 .bool_br_or, |                 .bool_br_or, | ||||||
|                 .bool_not, |                 .bool_not, | ||||||
|                 .call, |                 .call, | ||||||
|  |                 .field_call, | ||||||
|                 .cmp_lt, |                 .cmp_lt, | ||||||
|                 .cmp_lte, |                 .cmp_lte, | ||||||
|                 .cmp_eq, |                 .cmp_eq, | ||||||
| @ -1375,7 +1346,6 @@ pub const Inst = struct { | |||||||
|                 .field_ptr, |                 .field_ptr, | ||||||
|                 .field_ptr_init, |                 .field_ptr_init, | ||||||
|                 .field_val, |                 .field_val, | ||||||
|                 .field_call_bind, |  | ||||||
|                 .field_ptr_named, |                 .field_ptr_named, | ||||||
|                 .field_val_named, |                 .field_val_named, | ||||||
|                 .func, |                 .func, | ||||||
| @ -1427,16 +1397,11 @@ pub const Inst = struct { | |||||||
|                 .slice_start, |                 .slice_start, | ||||||
|                 .slice_end, |                 .slice_end, | ||||||
|                 .slice_sentinel, |                 .slice_sentinel, | ||||||
|  |                 .slice_length, | ||||||
|                 .import, |                 .import, | ||||||
|                 .typeof_log2_int_type, |                 .typeof_log2_int_type, | ||||||
|                 .switch_capture, |  | ||||||
|                 .switch_capture_ref, |  | ||||||
|                 .switch_capture_multi, |  | ||||||
|                 .switch_capture_multi_ref, |  | ||||||
|                 .switch_capture_tag, |  | ||||||
|                 .switch_block, |                 .switch_block, | ||||||
|                 .switch_cond, |                 .switch_block_ref, | ||||||
|                 .switch_cond_ref, |  | ||||||
|                 .array_base_ptr, |                 .array_base_ptr, | ||||||
|                 .field_base_ptr, |                 .field_base_ptr, | ||||||
|                 .struct_init_empty, |                 .struct_init_empty, | ||||||
| @ -1451,14 +1416,14 @@ pub const Inst = struct { | |||||||
|                 .union_init, |                 .union_init, | ||||||
|                 .field_type, |                 .field_type, | ||||||
|                 .field_type_ref, |                 .field_type_ref, | ||||||
|                 .int_to_enum, |                 .enum_from_int, | ||||||
|                 .enum_to_int, |                 .int_from_enum, | ||||||
|                 .type_info, |                 .type_info, | ||||||
|                 .size_of, |                 .size_of, | ||||||
|                 .bit_size_of, |                 .bit_size_of, | ||||||
|                 .ptr_to_int, |                 .int_from_ptr, | ||||||
|                 .align_of, |                 .align_of, | ||||||
|                 .bool_to_int, |                 .int_from_bool, | ||||||
|                 .embed_file, |                 .embed_file, | ||||||
|                 .error_name, |                 .error_name, | ||||||
|                 .sqrt, |                 .sqrt, | ||||||
| @ -1479,9 +1444,9 @@ pub const Inst = struct { | |||||||
|                 .type_name, |                 .type_name, | ||||||
|                 .frame_type, |                 .frame_type, | ||||||
|                 .frame_size, |                 .frame_size, | ||||||
|                 .float_to_int, |                 .int_from_float, | ||||||
|                 .int_to_float, |                 .float_from_int, | ||||||
|                 .int_to_ptr, |                 .ptr_from_int, | ||||||
|                 .float_cast, |                 .float_cast, | ||||||
|                 .int_cast, |                 .int_cast, | ||||||
|                 .ptr_cast, |                 .ptr_cast, | ||||||
| @ -1592,6 +1557,7 @@ pub const Inst = struct { | |||||||
|                 .check_comptime_control_flow = .un_node, |                 .check_comptime_control_flow = .un_node, | ||||||
|                 .for_len = .pl_node, |                 .for_len = .pl_node, | ||||||
|                 .call = .pl_node, |                 .call = .pl_node, | ||||||
|  |                 .field_call = .pl_node, | ||||||
|                 .cmp_lt = .pl_node, |                 .cmp_lt = .pl_node, | ||||||
|                 .cmp_lte = .pl_node, |                 .cmp_lte = .pl_node, | ||||||
|                 .cmp_eq = .pl_node, |                 .cmp_eq = .pl_node, | ||||||
| @ -1632,7 +1598,6 @@ pub const Inst = struct { | |||||||
|                 .field_val = .pl_node, |                 .field_val = .pl_node, | ||||||
|                 .field_ptr_named = .pl_node, |                 .field_ptr_named = .pl_node, | ||||||
|                 .field_val_named = .pl_node, |                 .field_val_named = .pl_node, | ||||||
|                 .field_call_bind = .pl_node, |  | ||||||
|                 .func = .pl_node, |                 .func = .pl_node, | ||||||
|                 .func_inferred = .pl_node, |                 .func_inferred = .pl_node, | ||||||
|                 .func_fancy = .pl_node, |                 .func_fancy = .pl_node, | ||||||
| @ -1664,6 +1629,7 @@ pub const Inst = struct { | |||||||
|                 .slice_start = .pl_node, |                 .slice_start = .pl_node, | ||||||
|                 .slice_end = .pl_node, |                 .slice_end = .pl_node, | ||||||
|                 .slice_sentinel = .pl_node, |                 .slice_sentinel = .pl_node, | ||||||
|  |                 .slice_length = .pl_node, | ||||||
|                 .store = .bin, |                 .store = .bin, | ||||||
|                 .store_node = .pl_node, |                 .store_node = .pl_node, | ||||||
|                 .store_to_block_ptr = .bin, |                 .store_to_block_ptr = .bin, | ||||||
| @ -1686,13 +1652,7 @@ pub const Inst = struct { | |||||||
|                 .err_union_code_ptr = .un_node, |                 .err_union_code_ptr = .un_node, | ||||||
|                 .enum_literal = .str_tok, |                 .enum_literal = .str_tok, | ||||||
|                 .switch_block = .pl_node, |                 .switch_block = .pl_node, | ||||||
|                 .switch_cond = .un_node, |                 .switch_block_ref = .pl_node, | ||||||
|                 .switch_cond_ref = .un_node, |  | ||||||
|                 .switch_capture = .switch_capture, |  | ||||||
|                 .switch_capture_ref = .switch_capture, |  | ||||||
|                 .switch_capture_multi = .switch_capture, |  | ||||||
|                 .switch_capture_multi_ref = .switch_capture, |  | ||||||
|                 .switch_capture_tag = .un_tok, |  | ||||||
|                 .array_base_ptr = .un_node, |                 .array_base_ptr = .un_node, | ||||||
|                 .field_base_ptr = .un_node, |                 .field_base_ptr = .un_node, | ||||||
|                 .validate_array_init_ty = .pl_node, |                 .validate_array_init_ty = .pl_node, | ||||||
| @ -1716,12 +1676,12 @@ pub const Inst = struct { | |||||||
|                 .size_of = .un_node, |                 .size_of = .un_node, | ||||||
|                 .bit_size_of = .un_node, |                 .bit_size_of = .un_node, | ||||||
| 
 | 
 | ||||||
|                 .ptr_to_int = .un_node, |                 .int_from_ptr = .un_node, | ||||||
|                 .compile_error = .un_node, |                 .compile_error = .un_node, | ||||||
|                 .set_eval_branch_quota = .un_node, |                 .set_eval_branch_quota = .un_node, | ||||||
|                 .enum_to_int = .un_node, |                 .int_from_enum = .un_node, | ||||||
|                 .align_of = .un_node, |                 .align_of = .un_node, | ||||||
|                 .bool_to_int = .un_node, |                 .int_from_bool = .un_node, | ||||||
|                 .embed_file = .un_node, |                 .embed_file = .un_node, | ||||||
|                 .error_name = .un_node, |                 .error_name = .un_node, | ||||||
|                 .panic = .un_node, |                 .panic = .un_node, | ||||||
| @ -1746,10 +1706,10 @@ pub const Inst = struct { | |||||||
|                 .frame_type = .un_node, |                 .frame_type = .un_node, | ||||||
|                 .frame_size = .un_node, |                 .frame_size = .un_node, | ||||||
| 
 | 
 | ||||||
|                 .float_to_int = .pl_node, |                 .int_from_float = .pl_node, | ||||||
|                 .int_to_float = .pl_node, |                 .float_from_int = .pl_node, | ||||||
|                 .int_to_ptr = .pl_node, |                 .ptr_from_int = .pl_node, | ||||||
|                 .int_to_enum = .pl_node, |                 .enum_from_int = .pl_node, | ||||||
|                 .float_cast = .pl_node, |                 .float_cast = .pl_node, | ||||||
|                 .int_cast = .pl_node, |                 .int_cast = .pl_node, | ||||||
|                 .ptr_cast = .pl_node, |                 .ptr_cast = .pl_node, | ||||||
| @ -1945,16 +1905,6 @@ pub const Inst = struct { | |||||||
|         /// The `@prefetch` builtin. |         /// The `@prefetch` builtin. | ||||||
|         /// `operand` is payload index to `BinNode`. |         /// `operand` is payload index to `BinNode`. | ||||||
|         prefetch, |         prefetch, | ||||||
|         /// Given a pointer to a struct or object that contains virtual fields, returns the |  | ||||||
|         /// named field.  If there is no named field, searches in the type for a decl that |  | ||||||
|         /// matches the field name.  The decl is resolved and we ensure that it's a function |  | ||||||
|         /// which can accept the object as the first parameter, with one pointer fixup.  If |  | ||||||
|         /// all of that works, this instruction produces a special "bound function" value |  | ||||||
|         /// which contains both the function and the saved first parameter value. |  | ||||||
|         /// Bound functions may only be used as the function parameter to a `call` or |  | ||||||
|         /// `builtin_call` instruction.  Any other use is invalid zir and may crash the compiler. |  | ||||||
|         /// Uses `pl_node` field. The AST node is the `@field` builtin. Payload is FieldNamedNode. |  | ||||||
|         field_call_bind_named, |  | ||||||
|         /// Implements the `@fence` builtin. |         /// Implements the `@fence` builtin. | ||||||
|         /// `operand` is payload index to `UnNode`. |         /// `operand` is payload index to `UnNode`. | ||||||
|         fence, |         fence, | ||||||
| @ -1980,10 +1930,10 @@ pub const Inst = struct { | |||||||
|         select, |         select, | ||||||
|         /// Implement builtin `@errToInt`. |         /// Implement builtin `@errToInt`. | ||||||
|         /// `operand` is payload index to `UnNode`. |         /// `operand` is payload index to `UnNode`. | ||||||
|         error_to_int, |         int_from_error, | ||||||
|         /// Implement builtin `@intToError`. |         /// Implement builtin `@errorFromInt`. | ||||||
|         /// `operand` is payload index to `UnNode`. |         /// `operand` is payload index to `UnNode`. | ||||||
|         int_to_error, |         error_from_int, | ||||||
|         /// Implement builtin `@Type`. |         /// Implement builtin `@Type`. | ||||||
|         /// `operand` is payload index to `UnNode`. |         /// `operand` is payload index to `UnNode`. | ||||||
|         /// `small` contains `NameStrategy`. |         /// `small` contains `NameStrategy`. | ||||||
| @ -2028,9 +1978,10 @@ pub const Inst = struct { | |||||||
|         /// Implements the `@inComptime` builtin. |         /// Implements the `@inComptime` builtin. | ||||||
|         /// `operand` is `src_node: i32`. |         /// `operand` is `src_node: i32`. | ||||||
|         in_comptime, |         in_comptime, | ||||||
|         /// Used as a placeholder for the capture of an `errdefer`. |         /// Used as a placeholder instruction which is just a dummy index for Sema to replace | ||||||
|         /// This is replaced by Sema with the captured value. |         /// with a specific value. For instance, this is used for the capture of an `errdefer`. | ||||||
|         errdefer_err_code, |         /// This should never appear in a body. | ||||||
|  |         value_placeholder, | ||||||
| 
 | 
 | ||||||
|         pub const InstData = struct { |         pub const InstData = struct { | ||||||
|             opcode: Extended, |             opcode: Extended, | ||||||
| @ -2058,107 +2009,89 @@ pub const Inst = struct { | |||||||
|     /// The tag type is specified so that it is safe to bitcast between `[]u32` |     /// The tag type is specified so that it is safe to bitcast between `[]u32` | ||||||
|     /// and `[]Ref`. |     /// and `[]Ref`. | ||||||
|     pub const Ref = enum(u32) { |     pub const Ref = enum(u32) { | ||||||
|         u1_type, |         u1_type = @intFromEnum(InternPool.Index.u1_type), | ||||||
|         u8_type, |         u8_type = @intFromEnum(InternPool.Index.u8_type), | ||||||
|         i8_type, |         i8_type = @intFromEnum(InternPool.Index.i8_type), | ||||||
|         u16_type, |         u16_type = @intFromEnum(InternPool.Index.u16_type), | ||||||
|         i16_type, |         i16_type = @intFromEnum(InternPool.Index.i16_type), | ||||||
|         u29_type, |         u29_type = @intFromEnum(InternPool.Index.u29_type), | ||||||
|         u32_type, |         u32_type = @intFromEnum(InternPool.Index.u32_type), | ||||||
|         i32_type, |         i32_type = @intFromEnum(InternPool.Index.i32_type), | ||||||
|         u64_type, |         u64_type = @intFromEnum(InternPool.Index.u64_type), | ||||||
|         i64_type, |         i64_type = @intFromEnum(InternPool.Index.i64_type), | ||||||
|         u128_type, |         u128_type = @intFromEnum(InternPool.Index.u128_type), | ||||||
|         i128_type, |         i128_type = @intFromEnum(InternPool.Index.i128_type), | ||||||
|         usize_type, |         usize_type = @intFromEnum(InternPool.Index.usize_type), | ||||||
|         isize_type, |         isize_type = @intFromEnum(InternPool.Index.isize_type), | ||||||
|         c_char_type, |         c_char_type = @intFromEnum(InternPool.Index.c_char_type), | ||||||
|         c_short_type, |         c_short_type = @intFromEnum(InternPool.Index.c_short_type), | ||||||
|         c_ushort_type, |         c_ushort_type = @intFromEnum(InternPool.Index.c_ushort_type), | ||||||
|         c_int_type, |         c_int_type = @intFromEnum(InternPool.Index.c_int_type), | ||||||
|         c_uint_type, |         c_uint_type = @intFromEnum(InternPool.Index.c_uint_type), | ||||||
|         c_long_type, |         c_long_type = @intFromEnum(InternPool.Index.c_long_type), | ||||||
|         c_ulong_type, |         c_ulong_type = @intFromEnum(InternPool.Index.c_ulong_type), | ||||||
|         c_longlong_type, |         c_longlong_type = @intFromEnum(InternPool.Index.c_longlong_type), | ||||||
|         c_ulonglong_type, |         c_ulonglong_type = @intFromEnum(InternPool.Index.c_ulonglong_type), | ||||||
|         c_longdouble_type, |         c_longdouble_type = @intFromEnum(InternPool.Index.c_longdouble_type), | ||||||
|         f16_type, |         f16_type = @intFromEnum(InternPool.Index.f16_type), | ||||||
|         f32_type, |         f32_type = @intFromEnum(InternPool.Index.f32_type), | ||||||
|         f64_type, |         f64_type = @intFromEnum(InternPool.Index.f64_type), | ||||||
|         f80_type, |         f80_type = @intFromEnum(InternPool.Index.f80_type), | ||||||
|         f128_type, |         f128_type = @intFromEnum(InternPool.Index.f128_type), | ||||||
|         anyopaque_type, |         anyopaque_type = @intFromEnum(InternPool.Index.anyopaque_type), | ||||||
|         bool_type, |         bool_type = @intFromEnum(InternPool.Index.bool_type), | ||||||
|         void_type, |         void_type = @intFromEnum(InternPool.Index.void_type), | ||||||
|         type_type, |         type_type = @intFromEnum(InternPool.Index.type_type), | ||||||
|         anyerror_type, |         anyerror_type = @intFromEnum(InternPool.Index.anyerror_type), | ||||||
|         comptime_int_type, |         comptime_int_type = @intFromEnum(InternPool.Index.comptime_int_type), | ||||||
|         comptime_float_type, |         comptime_float_type = @intFromEnum(InternPool.Index.comptime_float_type), | ||||||
|         noreturn_type, |         noreturn_type = @intFromEnum(InternPool.Index.noreturn_type), | ||||||
|         anyframe_type, |         anyframe_type = @intFromEnum(InternPool.Index.anyframe_type), | ||||||
|         empty_struct_literal, |         empty_struct_type = @intFromEnum(InternPool.Index.empty_struct_type), | ||||||
|         null_type, |         null_type = @intFromEnum(InternPool.Index.null_type), | ||||||
|         undefined_type, |         undefined_type = @intFromEnum(InternPool.Index.undefined_type), | ||||||
|         enum_literal_type, |         enum_literal_type = @intFromEnum(InternPool.Index.enum_literal_type), | ||||||
|         atomic_order_type, |         atomic_order_type = @intFromEnum(InternPool.Index.atomic_order_type), | ||||||
|         atomic_rmw_op_type, |         atomic_rmw_op_type = @intFromEnum(InternPool.Index.atomic_rmw_op_type), | ||||||
|         calling_convention_type, |         calling_convention_type = @intFromEnum(InternPool.Index.calling_convention_type), | ||||||
|         address_space_type, |         address_space_type = @intFromEnum(InternPool.Index.address_space_type), | ||||||
|         float_mode_type, |         float_mode_type = @intFromEnum(InternPool.Index.float_mode_type), | ||||||
|         reduce_op_type, |         reduce_op_type = @intFromEnum(InternPool.Index.reduce_op_type), | ||||||
|         modifier_type, |         call_modifier_type = @intFromEnum(InternPool.Index.call_modifier_type), | ||||||
|         prefetch_options_type, |         prefetch_options_type = @intFromEnum(InternPool.Index.prefetch_options_type), | ||||||
|         export_options_type, |         export_options_type = @intFromEnum(InternPool.Index.export_options_type), | ||||||
|         extern_options_type, |         extern_options_type = @intFromEnum(InternPool.Index.extern_options_type), | ||||||
|         type_info_type, |         type_info_type = @intFromEnum(InternPool.Index.type_info_type), | ||||||
|         manyptr_u8_type, |         manyptr_u8_type = @intFromEnum(InternPool.Index.manyptr_u8_type), | ||||||
|         manyptr_const_u8_type, |         manyptr_const_u8_type = @intFromEnum(InternPool.Index.manyptr_const_u8_type), | ||||||
|         fn_noreturn_no_args_type, |         manyptr_const_u8_sentinel_0_type = @intFromEnum(InternPool.Index.manyptr_const_u8_sentinel_0_type), | ||||||
|         fn_void_no_args_type, |         single_const_pointer_to_comptime_int_type = @intFromEnum(InternPool.Index.single_const_pointer_to_comptime_int_type), | ||||||
|         fn_naked_noreturn_no_args_type, |         slice_const_u8_type = @intFromEnum(InternPool.Index.slice_const_u8_type), | ||||||
|         fn_ccc_void_no_args_type, |         slice_const_u8_sentinel_0_type = @intFromEnum(InternPool.Index.slice_const_u8_sentinel_0_type), | ||||||
|         single_const_pointer_to_comptime_int_type, |         optional_noreturn_type = @intFromEnum(InternPool.Index.optional_noreturn_type), | ||||||
|         const_slice_u8_type, |         anyerror_void_error_union_type = @intFromEnum(InternPool.Index.anyerror_void_error_union_type), | ||||||
|         anyerror_void_error_union_type, |         generic_poison_type = @intFromEnum(InternPool.Index.generic_poison_type), | ||||||
|         generic_poison_type, |         unknown_type = @intFromEnum(InternPool.Index.unknown_type), | ||||||
|         unknown_type, |  | ||||||
| 
 | 
 | ||||||
|         /// `undefined` (untyped) |         undef = @intFromEnum(InternPool.Index.undefined_value), | ||||||
|         undef, |         zero = @intFromEnum(InternPool.Index.zero_comptime_int), | ||||||
|         /// `0` (comptime_int) |         zero_u8 = @intFromEnum(InternPool.Index.zero_u8), | ||||||
|         zero, |         one = @intFromEnum(InternPool.Index.one_comptime_int), | ||||||
|         /// `1` (comptime_int) |         one_u8 = @intFromEnum(InternPool.Index.one_u8), | ||||||
|         one, |         void_value = @intFromEnum(InternPool.Index.void_value), | ||||||
|         /// `{}` |         unreachable_value = @intFromEnum(InternPool.Index.unreachable_value), | ||||||
|         void_value, |         null_value = @intFromEnum(InternPool.Index.null_value), | ||||||
|         /// `unreachable` (noreturn type) |         bool_true = @intFromEnum(InternPool.Index.bool_true), | ||||||
|         unreachable_value, |         bool_false = @intFromEnum(InternPool.Index.bool_false), | ||||||
|         /// `null` (untyped) |         empty_aggregate = @intFromEnum(InternPool.Index.empty_aggregate), | ||||||
|         null_value, |         zero_usize = @intFromEnum(InternPool.Index.zero_usize), | ||||||
|         /// `true` |         one_usize = @intFromEnum(InternPool.Index.one_usize), | ||||||
|         bool_true, |         the_only_possible_value = @intFromEnum(InternPool.Index.the_only_possible_value), | ||||||
|         /// `false` |         generic_poison = @intFromEnum(InternPool.Index.generic_poison), | ||||||
|         bool_false, |         unknown_unknown = @intFromEnum(InternPool.Index.unknown_unknown), | ||||||
|         /// `.{}` (untyped) |  | ||||||
|         empty_struct, |  | ||||||
|         /// `0` (usize) |  | ||||||
|         zero_usize, |  | ||||||
|         /// `1` (usize) |  | ||||||
|         one_usize, |  | ||||||
|         /// `std.builtin.CallingConvention.C` |  | ||||||
|         calling_convention_c, |  | ||||||
|         /// `std.builtin.CallingConvention.Inline` |  | ||||||
|         calling_convention_inline, |  | ||||||
|         /// Used for generic parameters where the type and value |  | ||||||
|         /// is not known until generic function instantiation. |  | ||||||
|         generic_poison, |  | ||||||
|         unknown, |  | ||||||
| 
 | 
 | ||||||
|         ref_start_index, |         ref_start_index, | ||||||
| 
 |         none = @intFromEnum(InternPool.Index.none), | ||||||
|         /// This Ref does not correspond to any ZIR instruction or constant |  | ||||||
|         /// value and may instead be used as a sentinel to indicate null. |  | ||||||
|         none = std.math.maxInt(u32), |  | ||||||
|         _, |         _, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| @ -2290,10 +2223,6 @@ pub const Inst = struct { | |||||||
|             operand: Ref, |             operand: Ref, | ||||||
|             payload_index: u32, |             payload_index: u32, | ||||||
|         }, |         }, | ||||||
|         switch_capture: struct { |  | ||||||
|             switch_inst: Index, |  | ||||||
|             prong_index: u32, |  | ||||||
|         }, |  | ||||||
|         dbg_stmt: LineColumn, |         dbg_stmt: LineColumn, | ||||||
|         /// Used for unary operators which reference an inst, |         /// Used for unary operators which reference an inst, | ||||||
|         /// with an AST node source location. |         /// with an AST node source location. | ||||||
| @ -2363,7 +2292,6 @@ pub const Inst = struct { | |||||||
|             bool_br, |             bool_br, | ||||||
|             @"unreachable", |             @"unreachable", | ||||||
|             @"break", |             @"break", | ||||||
|             switch_capture, |  | ||||||
|             dbg_stmt, |             dbg_stmt, | ||||||
|             inst_node, |             inst_node, | ||||||
|             str_op, |             str_op, | ||||||
| @ -2579,6 +2507,19 @@ pub const Inst = struct { | |||||||
|         }; |         }; | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  |     /// Stored inside extra, with trailing arguments according to `args_len`. | ||||||
|  |     /// Implicit 0. arg_0_start: u32, // always same as `args_len` | ||||||
|  |     /// 1. arg_end: u32, // for each `args_len` | ||||||
|  |     /// arg_N_start is the same as arg_N-1_end | ||||||
|  |     pub const FieldCall = struct { | ||||||
|  |         // Note: Flags *must* come first so that unusedResultExpr | ||||||
|  |         // can find it when it goes to modify them. | ||||||
|  |         flags: Call.Flags, | ||||||
|  |         obj_ptr: Ref, | ||||||
|  |         /// Offset into `string_bytes`. | ||||||
|  |         field_name_start: u32, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     pub const TypeOfPeer = struct { |     pub const TypeOfPeer = struct { | ||||||
|         src_node: i32, |         src_node: i32, | ||||||
|         body_len: u32, |         body_len: u32, | ||||||
| @ -2658,6 +2599,14 @@ pub const Inst = struct { | |||||||
|         sentinel: Ref, |         sentinel: Ref, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  |     pub const SliceLength = struct { | ||||||
|  |         lhs: Ref, | ||||||
|  |         start: Ref, | ||||||
|  |         len: Ref, | ||||||
|  |         sentinel: Ref, | ||||||
|  |         start_src_node_offset: i32, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     /// The meaning of these operands depends on the corresponding `Tag`. |     /// The meaning of these operands depends on the corresponding `Tag`. | ||||||
|     pub const Bin = struct { |     pub const Bin = struct { | ||||||
|         lhs: Ref, |         lhs: Ref, | ||||||
| @ -2681,37 +2630,53 @@ pub const Inst = struct { | |||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     /// 0. multi_cases_len: u32 // If has_multi_cases is set. |     /// 0. multi_cases_len: u32 // If has_multi_cases is set. | ||||||
|     /// 1. else_body { // If has_else or has_under is set. |     /// 1. tag_capture_inst: u32 // If any_has_tag_capture is set. Index of instruction prongs use to refer to the inline tag capture. | ||||||
|     ///        body_len: u32, |     /// 2. else_body { // If has_else or has_under is set. | ||||||
|     ///        body member Index for every body_len |     ///        info: ProngInfo, | ||||||
|  |     ///        body member Index for every info.body_len | ||||||
|     ///     } |     ///     } | ||||||
|     /// 2. scalar_cases: { // for every scalar_cases_len |     /// 3. scalar_cases: { // for every scalar_cases_len | ||||||
|     ///        item: Ref, |     ///        item: Ref, | ||||||
|     ///        body_len: u32, |     ///        info: ProngInfo, | ||||||
|     ///        body member Index for every body_len |     ///        body member Index for every info.body_len | ||||||
|     ///     } |     ///     } | ||||||
|     /// 3. multi_cases: { // for every multi_cases_len |     /// 4. multi_cases: { // for every multi_cases_len | ||||||
|     ///        items_len: u32, |     ///        items_len: u32, | ||||||
|     ///        ranges_len: u32, |     ///        ranges_len: u32, | ||||||
|     ///        body_len: u32, |     ///        info: ProngInfo, | ||||||
|     ///        item: Ref // for every items_len |     ///        item: Ref // for every items_len | ||||||
|     ///        ranges: { // for every ranges_len |     ///        ranges: { // for every ranges_len | ||||||
|     ///            item_first: Ref, |     ///            item_first: Ref, | ||||||
|     ///            item_last: Ref, |     ///            item_last: Ref, | ||||||
|     ///        } |     ///        } | ||||||
|     ///        body member Index for every body_len |     ///        body member Index for every info.body_len | ||||||
|     ///    } |     ///    } | ||||||
|  |     /// | ||||||
|  |     /// When analyzing a case body, the switch instruction itself refers to the | ||||||
|  |     /// captured payload. Whether this is captured by reference or by value | ||||||
|  |     /// depends on whether the `byref` bit is set for the corresponding body. | ||||||
|     pub const SwitchBlock = struct { |     pub const SwitchBlock = struct { | ||||||
|         /// This is always a `switch_cond` or `switch_cond_ref` instruction. |         /// The operand passed to the `switch` expression. If this is a | ||||||
|         /// If it is a `switch_cond_ref` instruction, bits.is_ref is always true. |         /// `switch_block`, this is the operand value; if `switch_block_ref` it | ||||||
|         /// If it is a `switch_cond` instruction, bits.is_ref is always false. |         /// is a pointer to the operand. `switch_block_ref` is always used if | ||||||
|         /// Both `switch_cond` and `switch_cond_ref` return a value, not a pointer, |         /// any prong has a byref capture. | ||||||
|         /// that is useful for the case items, but cannot be used for capture values. |  | ||||||
|         /// For the capture values, Sema is expected to find the operand of this operand |  | ||||||
|         /// and use that. |  | ||||||
|         operand: Ref, |         operand: Ref, | ||||||
|         bits: Bits, |         bits: Bits, | ||||||
| 
 | 
 | ||||||
|  |         /// These are stored in trailing data in `extra` for each prong. | ||||||
|  |         pub const ProngInfo = packed struct(u32) { | ||||||
|  |             body_len: u28, | ||||||
|  |             capture: Capture, | ||||||
|  |             is_inline: bool, | ||||||
|  |             has_tag_capture: bool, | ||||||
|  | 
 | ||||||
|  |             pub const Capture = enum(u2) { | ||||||
|  |                 none, | ||||||
|  |                 by_val, | ||||||
|  |                 by_ref, | ||||||
|  |             }; | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|         pub const Bits = packed struct { |         pub const Bits = packed struct { | ||||||
|             /// If true, one or more prongs have multiple items. |             /// If true, one or more prongs have multiple items. | ||||||
|             has_multi_cases: bool, |             has_multi_cases: bool, | ||||||
| @ -2719,9 +2684,11 @@ pub const Inst = struct { | |||||||
|             has_else: bool, |             has_else: bool, | ||||||
|             /// If true, there is an underscore prong. This is mutually exclusive with `has_else`. |             /// If true, there is an underscore prong. This is mutually exclusive with `has_else`. | ||||||
|             has_under: bool, |             has_under: bool, | ||||||
|  |             /// If true, at least one prong has an inline tag capture. | ||||||
|  |             any_has_tag_capture: bool, | ||||||
|             scalar_cases_len: ScalarCasesLen, |             scalar_cases_len: ScalarCasesLen, | ||||||
| 
 | 
 | ||||||
|             pub const ScalarCasesLen = u29; |             pub const ScalarCasesLen = u28; | ||||||
| 
 | 
 | ||||||
|             pub fn specialProng(bits: Bits) SpecialProng { |             pub fn specialProng(bits: Bits) SpecialProng { | ||||||
|                 const has_else: u2 = @intFromBool(bits.has_else); |                 const has_else: u2 = @intFromBool(bits.has_else); | ||||||
| @ -2735,103 +2702,10 @@ pub const Inst = struct { | |||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         pub const ScalarProng = struct { |  | ||||||
|             item: Ref, |  | ||||||
|             body: []const Index, |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         /// TODO performance optimization: instead of having this helper method |  | ||||||
|         /// change the definition of switch_capture instruction to store extra_index |  | ||||||
|         /// instead of prong_index. This way, Sema won't be doing O(N^2) iterations |  | ||||||
|         /// over the switch prongs. |  | ||||||
|         pub fn getScalarProng( |  | ||||||
|             self: SwitchBlock, |  | ||||||
|             zir: Zir, |  | ||||||
|             extra_end: usize, |  | ||||||
|             prong_index: usize, |  | ||||||
|         ) ScalarProng { |  | ||||||
|             var extra_index: usize = extra_end; |  | ||||||
| 
 |  | ||||||
|             if (self.bits.has_multi_cases) { |  | ||||||
|                 extra_index += 1; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (self.bits.specialProng() != .none) { |  | ||||||
|                 const body_len = @truncate(u31, zir.extra[extra_index]); |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const body = zir.extra[extra_index..][0..body_len]; |  | ||||||
|                 extra_index += body.len; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var scalar_i: usize = 0; |  | ||||||
|             while (true) : (scalar_i += 1) { |  | ||||||
|                 const item = @enumFromInt(Ref, zir.extra[extra_index]); |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const body_len = @truncate(u31, zir.extra[extra_index]); |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const body = zir.extra[extra_index..][0..body_len]; |  | ||||||
|                 extra_index += body.len; |  | ||||||
| 
 |  | ||||||
|                 if (scalar_i < prong_index) continue; |  | ||||||
| 
 |  | ||||||
|                 return .{ |  | ||||||
|                     .item = item, |  | ||||||
|                     .body = body, |  | ||||||
|                 }; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         pub const MultiProng = struct { |         pub const MultiProng = struct { | ||||||
|             items: []const Ref, |             items: []const Ref, | ||||||
|             body: []const Index, |             body: []const Index, | ||||||
|         }; |         }; | ||||||
| 
 |  | ||||||
|         pub fn getMultiProng( |  | ||||||
|             self: SwitchBlock, |  | ||||||
|             zir: Zir, |  | ||||||
|             extra_end: usize, |  | ||||||
|             prong_index: usize, |  | ||||||
|         ) MultiProng { |  | ||||||
|             // +1 for self.bits.has_multi_cases == true |  | ||||||
|             var extra_index: usize = extra_end + 1; |  | ||||||
| 
 |  | ||||||
|             if (self.bits.specialProng() != .none) { |  | ||||||
|                 const body_len = @truncate(u31, zir.extra[extra_index]); |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const body = zir.extra[extra_index..][0..body_len]; |  | ||||||
|                 extra_index += body.len; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var scalar_i: usize = 0; |  | ||||||
|             while (scalar_i < self.bits.scalar_cases_len) : (scalar_i += 1) { |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const body_len = @truncate(u31, zir.extra[extra_index]); |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 extra_index += body_len; |  | ||||||
|             } |  | ||||||
|             var multi_i: u32 = 0; |  | ||||||
|             while (true) : (multi_i += 1) { |  | ||||||
|                 const items_len = zir.extra[extra_index]; |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const ranges_len = zir.extra[extra_index]; |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const body_len = @truncate(u31, zir.extra[extra_index]); |  | ||||||
|                 extra_index += 1; |  | ||||||
|                 const items = zir.refSlice(extra_index, items_len); |  | ||||||
|                 extra_index += items_len; |  | ||||||
|                 // Each range has a start and an end. |  | ||||||
|                 extra_index += 2 * ranges_len; |  | ||||||
| 
 |  | ||||||
|                 const body = zir.extra[extra_index..][0..body_len]; |  | ||||||
|                 extra_index += body_len; |  | ||||||
| 
 |  | ||||||
|                 if (multi_i < prong_index) continue; |  | ||||||
|                 return .{ |  | ||||||
|                     .items = items, |  | ||||||
|                     .body = body, |  | ||||||
|                 }; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     pub const Field = struct { |     pub const Field = struct { | ||||||
| @ -2845,12 +2719,6 @@ pub const Inst = struct { | |||||||
|         field_name: Ref, |         field_name: Ref, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     pub const FieldNamedNode = struct { |  | ||||||
|         node: i32, |  | ||||||
|         lhs: Ref, |  | ||||||
|         field_name: Ref, |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     pub const As = struct { |     pub const As = struct { | ||||||
|         dest_type: Ref, |         dest_type: Ref, | ||||||
|         operand: Ref, |         operand: Ref, | ||||||
| @ -3832,6 +3700,7 @@ pub fn indexToRef(inst: Inst.Index) Inst.Ref { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub fn refToIndex(inst: Inst.Ref) ?Inst.Index { | pub fn refToIndex(inst: Inst.Ref) ?Inst.Index { | ||||||
|  |     assert(inst != .none); | ||||||
|     const ref_int = @intFromEnum(inst); |     const ref_int = @intFromEnum(inst); | ||||||
|     if (ref_int >= ref_start_index) { |     if (ref_int >= ref_start_index) { | ||||||
|         return ref_int - ref_start_index; |         return ref_int - ref_start_index; | ||||||
| @ -3839,3 +3708,8 @@ pub fn refToIndex(inst: Inst.Ref) ?Inst.Index { | |||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | pub fn refToIndexAllowNone(inst: Inst.Ref) ?Inst.Index { | ||||||
|  |     if (inst == .none) return null; | ||||||
|  |     return refToIndex(inst); | ||||||
|  | } | ||||||
|  | |||||||
| @ -236,12 +236,10 @@ pub fn translate( | |||||||
|                 const body_size = @sizeOf(std.zig.Server.Message.EmitBinPath); |                 const body_size = @sizeOf(std.zig.Server.Message.EmitBinPath); | ||||||
|                 if (header.bytes_len <= body_size) return error.InvalidResponse; |                 if (header.bytes_len <= body_size) return error.InvalidResponse; | ||||||
| 
 | 
 | ||||||
|                 const trailing_size = header.bytes_len - body_size; |  | ||||||
| 
 |  | ||||||
|                 _ = try zcs.receiveEmitBinPath(); |                 _ = try zcs.receiveEmitBinPath(); | ||||||
| 
 | 
 | ||||||
|                 const result_path = try zcs.receiveBytes(allocator, trailing_size); |                 const trailing_size = header.bytes_len - body_size; | ||||||
|                 defer allocator.free(result_path); |                 const result_path = zcs.pooler.fifo(.in).readableSliceOfLen(trailing_size); | ||||||
| 
 | 
 | ||||||
|                 return Result{ .success = try URI.fromPath(allocator, std.mem.sliceTo(result_path, '\n')) }; |                 return Result{ .success = try URI.fromPath(allocator, std.mem.sliceTo(result_path, '\n')) }; | ||||||
|             }, |             }, | ||||||
|  | |||||||
| @ -509,6 +509,17 @@ test "completion - block" { | |||||||
|     , &.{ |     , &.{ | ||||||
|         .{ .label = "blk", .kind = .Text }, // idk what kind this should be |         .{ .label = "blk", .kind = .Text }, // idk what kind this should be | ||||||
|     }); |     }); | ||||||
|  | 
 | ||||||
|  |     try testCompletion( | ||||||
|  |         \\const S = struct { alpha: u32 }; | ||||||
|  |         \\const foo: S = undefined; | ||||||
|  |         \\const bar = blk: { | ||||||
|  |         \\    break :blk foo; | ||||||
|  |         \\}; | ||||||
|  |         \\const baz = bar.<cursor> | ||||||
|  |     , &.{ | ||||||
|  |         .{ .label = "alpha", .kind = .Field, .detail = "alpha: u32" }, | ||||||
|  |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn testCompletion(source: []const u8, expected_completions: []const Completion) !void { | fn testCompletion(source: []const u8, expected_completions: []const Completion) !void { | ||||||
|  | |||||||
| @ -48,6 +48,23 @@ test "inlayhints - function self parameter" { | |||||||
|         \\const foo: Foo = .{}; |         \\const foo: Foo = .{}; | ||||||
|         \\const _ = foo.bar(<alpha>5,<beta>""); |         \\const _ = foo.bar(<alpha>5,<beta>""); | ||||||
|     ); |     ); | ||||||
|  |     try testInlayHints( | ||||||
|  |         \\const Foo = struct { pub fn bar(self: Foo, alpha: u32, beta: anytype) void {} }; | ||||||
|  |         \\const foo: Foo = .{}; | ||||||
|  |         \\const _ = foo.bar(<alpha>5,<beta>4); | ||||||
|  |     ); | ||||||
|  |     try testInlayHints( | ||||||
|  |         \\const Foo = struct { pub fn bar(self: Foo, alpha: u32, beta: []const u8) void {} }; | ||||||
|  |         \\const _ = Foo.bar(<self>undefined,<alpha>5,<beta>""); | ||||||
|  |     ); | ||||||
|  |     try testInlayHints( | ||||||
|  |         \\const Foo = struct { | ||||||
|  |         \\  pub fn bar(self: Foo, alpha: u32, beta: []const u8) void {} | ||||||
|  |         \\  pub fn foo() void { | ||||||
|  |         \\      bar(<self>undefined,<alpha>5,<beta>""); | ||||||
|  |         \\  } | ||||||
|  |         \\}; | ||||||
|  |     ); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| test "inlayhints - builtin call" { | test "inlayhints - builtin call" { | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user