update translate-c to latest

upstream commit 46b5609b5ac4c0a896217d1d984f3ae50e4810b5
This commit is contained in:
Andrew Kelley
2026-04-11 15:05:16 -07:00
parent 67c6ac947a
commit 5149128d22
7 changed files with 596 additions and 188 deletions
+80 -11
View File
@@ -266,7 +266,8 @@ fn parseCNumLit(mt: *MacroTranslator) ParseError!ZigNode {
const lit_bytes = mt.tokSlice();
mt.i += 1;
var bytes = try std.ArrayList(u8).initCapacity(arena, lit_bytes.len + 3);
// +3 for prefix and +2 for suffix
var bytes = try std.ArrayList(u8).initCapacity(arena, lit_bytes.len + 3 + 2);
const prefix = aro.Tree.Token.NumberPrefix.fromString(lit_bytes);
switch (prefix) {
@@ -350,13 +351,21 @@ fn parseCNumLit(mt: *MacroTranslator) ParseError!ZigNode {
if (is_float) {
const type_node = try ZigTag.type.create(arena, switch (suffix) {
.F16 => "f16",
.F => "f32",
.None => "f64",
.L => "c_longdouble",
.F, .F32 => "f32",
.None, .F32x, .F64 => "f64",
.L, .F64x => "c_longdouble",
.W => "f80",
.Q, .F128 => "f128",
else => unreachable,
else => {
try mt.fail("TODO: float literal suffix: '{s}'", .{suffix_str});
return error.ParseError;
},
});
if (bytes.getLast() == '.') {
bytes.appendAssumeCapacity('0');
} else if (mem.findAny(u8, bytes.items, ".eEpP") == null) {
bytes.appendSliceAssumeCapacity(".0");
}
const rhs = try ZigTag.float_literal.create(arena, bytes.items);
return ZigTag.as.create(arena, .{ .lhs = type_node, .rhs = rhs });
} else {
@@ -582,6 +591,7 @@ fn escapeUnprintables(mt: *MacroTranslator) ![]const u8 {
fn parseCPrimaryExpr(mt: *MacroTranslator, scope: *Scope) ParseError!ZigNode {
const arena = mt.t.arena;
const gpa = mt.t.gpa;
const tok = mt.peek();
switch (tok) {
.char_literal,
@@ -646,6 +656,51 @@ fn parseCPrimaryExpr(mt: *MacroTranslator, scope: *Scope) ParseError!ZigNode {
}
return identifier;
},
.keyword_generic => {
mt.i += 1;
try mt.expect(.l_paren);
const param = try mt.parseCCondExpr(scope);
const typeof_param = try ZigTag.typeof.create(arena, param);
try mt.expect(.comma);
var cases: std.ArrayList(ZigNode) = .empty;
defer cases.deinit(gpa);
var has_default = false;
while (true) {
const case = if (mt.eat(.keyword_default)) blk: {
has_default = true;
try mt.expect(.colon);
const expr = try mt.parseCCondExpr(scope);
break :blk try ZigTag.switch_else.create(arena, expr);
} else blk: {
const case_type = try mt.parseCTypeName(scope) orelse {
try mt.fail("unable to translate C expr: expected type instead got '{s}'", .{mt.peek().symbol()});
return error.ParseError;
};
try mt.expect(.colon);
const expr = try mt.parseCCondExpr(scope);
break :blk try ZigTag.switch_prong.create(arena, .{
.cases = try arena.dupe(ZigNode, &.{case_type}),
.cond = expr,
});
};
try cases.append(gpa, case);
if (!mt.eat(.comma)) break;
}
try mt.expect(.r_paren);
if (!has_default) try cases.append(gpa, try ZigTag.switch_else.create(
arena,
try ZigTag.@"comptime".create(arena, ZigTag.@"unreachable".init()),
));
const sw = try ZigTag.@"switch".create(arena, .{
.cond = typeof_param,
.cases = try arena.dupe(ZigNode, cases.items),
});
return sw;
},
else => {},
}
@@ -678,8 +733,10 @@ fn macroIntToBool(mt: *MacroTranslator, node: ZigNode) !ZigNode {
}
fn parseCCondExpr(mt: *MacroTranslator, scope: *Scope) ParseError!ZigNode {
const node = try mt.parseCOrExpr(scope);
if (!mt.eat(.question_mark)) return node;
const condition = try mt.parseCOrExpr(scope);
if (!mt.eat(.question_mark)) return condition;
const bool_ty = try ZigTag.type.create(mt.t.arena, "bool");
const node = try mt.t.createHelperCallNode(.cast, &.{ bool_ty, condition });
const then_body = try mt.parseCOrExpr(scope);
try mt.expect(.colon);
@@ -1135,6 +1192,8 @@ fn parseCPostfixExpr(mt: *MacroTranslator, scope: *Scope, type_name: ?ZigNode) P
.string_literal_utf_8,
.string_literal_utf_32,
.string_literal_wide,
.macro_param,
.macro_param_no_expand,
=> {},
.identifier, .extended_identifier => {
if (mt.t.global_scope.blank_macros.contains(mt.tokSlice())) {
@@ -1160,8 +1219,13 @@ fn parseCPostfixExprInner(mt: *MacroTranslator, scope: *Scope, type_name: ?ZigNo
mt.i += 1;
const tok = mt.tokens[mt.i];
if (tok.id == .macro_param or tok.id == .macro_param_no_expand) {
try mt.fail("unable to translate C expr: field access using macro parameter", .{});
return error.ParseError;
const param = mt.macro.params[tok.end];
mt.i += 1;
const mangled_name = scope.getAlias(param) orelse param;
const field_name = try ZigTag.identifier.create(arena, mangled_name);
node = try ZigTag.field_builtin.create(arena, .{ .lhs = node, .rhs = field_name });
continue;
}
const field_name = mt.tokSlice();
try mt.expect(.identifier);
@@ -1172,8 +1236,13 @@ fn parseCPostfixExprInner(mt: *MacroTranslator, scope: *Scope, type_name: ?ZigNo
mt.i += 1;
const tok = mt.tokens[mt.i];
if (tok.id == .macro_param or tok.id == .macro_param_no_expand) {
try mt.fail("unable to translate C expr: field access using macro parameter", .{});
return error.ParseError;
const param = mt.macro.params[tok.end];
mt.i += 1;
const mangled_name = scope.getAlias(param) orelse param;
const field_name = try ZigTag.identifier.create(arena, mangled_name);
node = try ZigTag.field_builtin.create(arena, .{ .lhs = node, .rhs = field_name });
continue;
}
const field_name = mt.tokSlice();
try mt.expect(.identifier);
+1 -9
View File
@@ -65,10 +65,7 @@ const templates = [_]Template{
.{ "CAST_OR_CALL(X, Y) ((X)(Y))", .CAST_OR_CALL },
.{
\\wl_container_of(ptr, sample, member) \
\\(__typeof__(sample))((char *)(ptr) - \
\\ offsetof(__typeof__(*sample), member))
,
"wl_container_of(ptr, sample, member) (__typeof__(sample))((char *)(ptr) - offsetof(__typeof__(*sample), member))",
.WL_CONTAINER_OF,
},
@@ -267,11 +264,6 @@ test "Macro matching" {
try helper.checkMacro(allocator, pattern_list, "BAR(Z) (Z ## LL)", .LL_SUFFIX);
try helper.checkMacro(allocator, pattern_list, "BAR(Z) (Z ## UL)", .UL_SUFFIX);
try helper.checkMacro(allocator, pattern_list, "BAR(Z) (Z ## ULL)", .ULL_SUFFIX);
try helper.checkMacro(allocator, pattern_list,
\\container_of(a, b, c) \
\\(__typeof__(b))((char *)(a) - \
\\ offsetof(__typeof__(*b), c))
, .WL_CONTAINER_OF);
try helper.checkMacro(allocator, pattern_list, "NO_MATCH(X, Y) (X + Y)", null);
try helper.checkMacro(allocator, pattern_list, "CAST_OR_CALL(X, Y) (X)(Y)", .CAST_OR_CALL);
+43 -6
View File
@@ -18,7 +18,22 @@ pub const ContainerMemberFns = struct {
container_decl_ptr: *ast.Node,
member_fns: std.ArrayList(*ast.Payload.Func) = .empty,
};
pub const ContainerMemberFnsHashMap = std.AutoArrayHashMapUnmanaged(aro.QualType, ContainerMemberFns);
pub const ContainerMemberFnsHashMap = std.ArrayHashMapUnmanaged(
aro.QualType,
ContainerMemberFns,
struct {
pub fn hash(self: @This(), key: aro.QualType) u32 {
const auto_hash = std.array_hash_map.getAutoHashFn(aro.QualType, @This());
return auto_hash(self, key.unqualified());
}
pub fn eql(self: @This(), a: aro.QualType, b: aro.QualType, b_index: usize) bool {
const auto_eql = std.array_hash_map.getAutoEqlFn(aro.QualType, @This());
return auto_eql(self, a.unqualified(), b.unqualified(), b_index);
}
},
false,
);
id: Id,
parent: ?*Scope,
@@ -254,7 +269,12 @@ pub const Root = struct {
var member_names: std.StringArrayHashMapUnmanaged(void) = .empty;
defer member_names.deinit(gpa);
for (root.container_member_fns_map.values()) |members| {
for (root.container_member_fns_map.keys(), root.container_member_fns_map.values()) |container_qt, members| {
// Get the container name
const container_name = root.translator.unnamed_typedefs.get(container_qt) orelse
container_qt.getRecord(root.translator.comp).?.name.lookup(root.translator.comp);
std.debug.assert(container_name.len > 0);
member_names.clearRetainingCapacity();
const decls_ptr = switch (members.container_decl_ptr.tag()) {
.@"struct", .@"union" => blk_record: {
@@ -274,7 +294,7 @@ pub const Root = struct {
members.container_decl_ptr.* = container_decl;
break :blk_opaque &container_decl.castTag(.@"opaque").?.data.decls;
},
else => return,
else => continue,
};
const old_decls = decls_ptr.*;
@@ -299,9 +319,26 @@ pub const Root = struct {
for (members.member_fns.items) |func| {
const func_name = func.data.name.?;
const func_name_trimmed = std.mem.trimEnd(u8, func_name, "_");
const last_idx = std.mem.findLast(u8, func_name_trimmed, "_") orelse continue;
const func_name_alias = func_name[last_idx + 1 ..];
const func_name_alias = blk: {
// Try multiple candidate prefixes to extract the alias
// 1. typedef struct { ... } foo; -> foo_get_bar() extracts "get_bar"
// 2. typedef struct _foo foo; -> foo_get_bar() extracts "get_bar"
const container_name_trimmed = std.mem.trimStart(u8, container_name, "_");
const suffix = std.mem.cutPrefix(u8, func_name, container_name_trimmed);
// Check suffix starts with '_' to avoid invalid aliases like "1_get_bar" from foo1_get_bar()
if (suffix) |alias| if (alias.len > 0 and alias[0] == '_') {
const alias_trimmed = std.mem.trimStart(u8, alias, "_");
if (alias_trimmed.len > 0) break :blk alias_trimmed;
};
// Doesn't match any prefix - fallback to trimming trailing underscores and using last segment
const func_name_trimmed = std.mem.trimEnd(u8, func_name, "_");
const last_idx = std.mem.findLast(u8, func_name_trimmed, "_") orelse continue;
break :blk func_name[last_idx + 1 ..];
};
// Skip if the alias conflicts with an existing type
if (root.contains(func_name_alias)) continue;
const member_name_slot = try member_names.getOrPutValue(gpa, func_name_alias, {});
if (member_name_slot.found_existing) continue;
func_ref_vars[count] = try ast.Node.Tag.pub_var_simple.create(arena, .{
+364 -126
View File
@@ -19,10 +19,63 @@ const MacroTranslator = @import("MacroTranslator.zig");
const PatternList = @import("PatternList.zig");
const Scope = @import("Scope.zig");
const AnonymousRecordFieldNames = struct {
pub const Key = struct {
parent: QualType,
field: QualType,
};
pub const Context = struct {
pub fn hash(ctx: Context, key: Key) u64 {
const auto_hash = std.hash_map.getAutoHashFn(Key, Context);
return auto_hash(ctx, .{
.parent = key.parent.unqualified(),
.field = key.field.unqualified(),
});
}
pub fn eql(ctx: Context, a: Key, b: Key) bool {
const auto_eql = std.hash_map.getAutoEqlFn(Key, Context);
return auto_eql(ctx, .{
.parent = a.parent.unqualified(),
.field = a.field.unqualified(),
}, .{
.parent = b.parent.unqualified(),
.field = b.field.unqualified(),
});
}
};
};
pub const QualTypeHashContext = struct {
pub fn hash(ctx: QualTypeHashContext, key: QualType) u64 {
const auto_hash = std.hash_map.getAutoHashFn(QualType, QualTypeHashContext);
return auto_hash(ctx, key.unqualified());
}
pub fn eql(ctx: QualTypeHashContext, a: QualType, b: QualType) bool {
const auto_eql = std.hash_map.getAutoEqlFn(QualType, QualTypeHashContext);
return auto_eql(ctx, a.unqualified(), b.unqualified());
}
};
pub const Error = std.mem.Allocator.Error;
pub const MacroProcessingError = Error || error{UnexpectedMacroToken};
pub const TypeError = Error || error{UnsupportedType};
pub const TransError = TypeError || error{UnsupportedTranslation};
pub const TransError = TypeError || error{ UnsupportedTranslation, SelfReferential };
/// Control when to treat a trailing array as a flexible array member.
/// Mirrors the -fstrict-flex-arrays=<n> compiler flag.
pub const StrictFlexArraysLevel = enum {
/// Any trailing array member is a flexible array.
@"0",
/// Trailing arrays of size 0, 1, or undefined are flexible.
@"1",
/// Trailing arrays of size 0 or undefined are flexible (default).
@"2",
/// Only trailing arrays of undefined size are flexible.
@"3",
};
const Translator = @This();
@@ -33,6 +86,17 @@ comp: *aro.Compilation,
/// The Preprocessor that produced the source for `tree`.
pp: *const aro.Preprocessor,
/// Should static functions be translated as `pub`.
pub_static: bool,
/// Should function bodies be translated.
func_bodies: bool,
/// Should macro names of literals be preserved.
keep_macro_literals: bool,
/// Should struct fields be default initialized.
default_init: bool,
/// Control when to treat a trailing array as a flexible array member.
strict_flex_arrays: StrictFlexArraysLevel,
gpa: mem.Allocator,
arena: mem.Allocator,
@@ -44,14 +108,16 @@ mangle_count: u32 = 0,
/// Table of declarations for enum, struct, union and typedef types.
type_decls: std.AutoArrayHashMapUnmanaged(Node.Index, []const u8) = .empty,
/// Table of record decls that have been demoted to opaques.
opaque_demotes: std.AutoHashMapUnmanaged(QualType, void) = .empty,
opaque_demotes: std.HashMapUnmanaged(QualType, void, QualTypeHashContext, std.hash_map.default_max_load_percentage) = .empty,
/// Table of unnamed enums and records that are child types of typedefs.
unnamed_typedefs: std.AutoHashMapUnmanaged(QualType, []const u8) = .empty,
unnamed_typedefs: std.HashMapUnmanaged(QualType, []const u8, QualTypeHashContext, std.hash_map.default_max_load_percentage) = .empty,
/// Table of anonymous record to generated field names.
anonymous_record_field_names: std.AutoHashMapUnmanaged(struct {
parent: QualType,
field: QualType,
}, []const u8) = .empty,
anonymous_record_field_names: std.HashMapUnmanaged(
AnonymousRecordFieldNames.Key,
[]const u8,
AnonymousRecordFieldNames.Context,
std.hash_map.default_max_load_percentage,
) = .empty,
/// This one is different than the root scope's name table. This contains
/// a list of names that we found by visiting all the top level decls without
@@ -75,6 +141,10 @@ typedefs: std.StringArrayHashMapUnmanaged(void) = .empty,
/// The lhs lval of a compound assignment expression.
compound_assign_dummy: ?ZigNode = null,
/// Set of variables whose initializers are currently being translated.
/// Used to detect self-referential initializers.
wip_var_inits: std.AutoHashMapUnmanaged(Node.Index, void) = .empty,
pub fn getMangle(t: *Translator) u32 {
t.mangle_count += 1;
return t.mangle_count;
@@ -98,10 +168,9 @@ fn maybeSuppressResult(t: *Translator, used: ResultUsed, result: ZigNode) TransE
pub fn addTopLevelDecl(t: *Translator, name: []const u8, decl_node: ZigNode) !void {
const gop = try t.global_scope.sym_table.getOrPut(t.gpa, name);
if (!gop.found_existing) {
gop.value_ptr.* = decl_node;
try t.global_scope.nodes.append(t.gpa, decl_node);
}
if (gop.found_existing) return; // Any duplicate decls are equivalent
gop.value_ptr.* = decl_node;
try t.global_scope.nodes.append(t.gpa, decl_node);
}
fn fail(
@@ -172,6 +241,12 @@ pub const Options = struct {
comp: *aro.Compilation,
pp: *const aro.Preprocessor,
tree: *const aro.Tree,
module_libs: bool,
pub_static: bool,
func_bodies: bool,
keep_macro_literals: bool,
default_init: bool,
strict_flex_arrays: StrictFlexArraysLevel,
};
pub fn translate(options: Options) mem.Allocator.Error![]u8 {
@@ -188,6 +263,11 @@ pub fn translate(options: Options) mem.Allocator.Error![]u8 {
.comp = options.comp,
.pp = options.pp,
.tree = options.tree,
.pub_static = options.pub_static,
.func_bodies = options.func_bodies,
.keep_macro_literals = options.keep_macro_literals,
.default_init = options.default_init,
.strict_flex_arrays = options.strict_flex_arrays,
};
translator.global_scope.* = Scope.Root.init(&translator);
defer {
@@ -200,6 +280,7 @@ pub fn translate(options: Options) mem.Allocator.Error![]u8 {
translator.anonymous_record_field_names.deinit(gpa);
translator.typedefs.deinit(gpa);
translator.global_scope.deinit();
translator.wip_var_inits.deinit(gpa);
}
try translator.prepopulateGlobalNameTable();
@@ -227,7 +308,6 @@ pub fn translate(options: Options) mem.Allocator.Error![]u8 {
\\pub const __builtin = @import("std").zig.c_translation.builtins;
\\pub const __helpers = @import("std").zig.c_translation.helpers;
\\
\\
) catch return error.OutOfMemory;
var zig_ast = try ast.render(gpa, translator.global_scope.nodes.items);
@@ -261,10 +341,12 @@ fn prepopulateGlobalNameTable(t: *Translator) !void {
const gop = try t.unnamed_typedefs.getOrPut(t.gpa, base.qt);
if (gop.found_existing) {
// One typedef can declare multiple names.
// TODO Don't put this one in `decl_table` so it's processed later.
// Don't put this one in `decl_table` so it's processed later.
continue;
}
gop.value_ptr.* = decl_name;
try t.type_decls.put(t.gpa, decl, decl_name);
try t.typedefs.put(t.gpa, decl_name, {});
},
.struct_decl,
@@ -344,15 +426,26 @@ fn transDecl(t: *Translator, scope: *Scope, decl: Node.Index) !void {
try t.transRecordDecl(scope, record_decl.container_qt);
},
.struct_forward_decl, .union_forward_decl => |record_decl| {
if (record_decl.definition) |some| {
return t.transDecl(scope, some);
}
try t.transRecordDecl(scope, record_decl.container_qt);
},
.enum_decl => |enum_decl| {
try t.transEnumDecl(scope, enum_decl.container_qt);
},
.enum_forward_decl => |enum_decl| {
if (enum_decl.definition) |some| {
return t.transDecl(scope, some);
}
try t.transEnumDecl(scope, enum_decl.container_qt);
},
.enum_field,
.record_field,
.struct_forward_decl,
.union_forward_decl,
.enum_forward_decl,
=> return,
.function => |function| {
@@ -364,7 +457,7 @@ fn transDecl(t: *Translator, scope: *Scope, decl: Node.Index) !void {
.variable => |variable| {
if (variable.definition != null) return;
try t.transVarDecl(scope, variable);
try t.transVarDecl(scope, variable, decl);
},
.static_assert => |static_assert| {
try t.transStaticAssert(&t.global_scope.base, static_assert);
@@ -531,13 +624,6 @@ fn transRecordDecl(t: *Translator, scope: *Scope, record_qt: QualType) Error!voi
break :init ZigTag.opaque_literal.init();
}
// Demote record to opaque if it contains an opaque field
if (t.typeWasDemotedToOpaque(field.qt)) {
try t.opaque_demotes.put(t.gpa, base.qt, {});
try t.warn(scope, field_loc, "{s} demoted to opaque type - has opaque field", .{container_kind_name});
break :init ZigTag.opaque_literal.init();
}
var field_name = field.name.lookup(t.comp);
if (field.name_tok == 0) {
field_name = try std.fmt.allocPrint(t.arena, "unnamed_{d}", .{unnamed_field_count});
@@ -548,23 +634,22 @@ fn transRecordDecl(t: *Translator, scope: *Scope, record_qt: QualType) Error!voi
}, field_name);
}
const field_alignment = if (has_alignment_attributes)
t.alignmentForField(record_ty, head_field_alignment, field_index)
else
null;
const field_type = field_type: {
// Check if this is a flexible array member.
flexible: {
if (field_index != record_ty.fields.len - 1 and container_kind != .@"union") break :flexible;
const array_ty = field.qt.get(t.comp, .array) orelse break :flexible;
if (array_ty.len != .incomplete and (array_ty.len != .fixed or array_ty.len.fixed != 0)) break :flexible;
if (!t.isFlexibleArrayLen(array_ty.len)) break :flexible;
const elem_type = t.transType(scope, array_ty.elem, field_loc) catch |err| switch (err) {
error.UnsupportedType => break :flexible,
else => |e| return e,
};
const zero_array = try ZigTag.array_type.create(t.arena, .{ .len = 0, .elem_type = elem_type });
const backing_array_len: usize = switch (array_ty.len) {
.fixed => |n| @intCast(n),
else => 0,
};
const backing_array = try ZigTag.array_type.create(t.arena, .{ .len = backing_array_len, .elem_type = elem_type });
const member_name = field_name;
field_name = try std.fmt.allocPrint(t.arena, "_{s}", .{field_name});
@@ -572,7 +657,7 @@ fn transRecordDecl(t: *Translator, scope: *Scope, record_qt: QualType) Error!voi
const member = try t.createFlexibleMemberFn(member_name, field_name);
try functions.append(t.gpa, member);
break :field_type zero_array;
break :field_type backing_array;
}
break :field_type t.transType(scope, field.qt, field_loc) catch |err| switch (err) {
@@ -588,10 +673,22 @@ fn transRecordDecl(t: *Translator, scope: *Scope, record_qt: QualType) Error!voi
};
};
// Demote record to opaque if it contains an opaque field
if (t.typeWasDemotedToOpaque(field.qt)) {
try t.opaque_demotes.put(t.gpa, base.qt, {});
try t.warn(scope, field_loc, "{s} demoted to opaque type - has opaque field", .{container_kind_name});
break :init ZigTag.opaque_literal.init();
}
const field_alignment = if (has_alignment_attributes)
t.alignmentForField(record_ty, head_field_alignment, field_index)
else
null;
// C99 introduced designated initializers for structs. Omitted fields are implicitly
// initialized to zero. Some C APIs are designed with this in mind. Defaulting to zero
// values for translated struct fields permits Zig code to comfortably use such an API.
const default_value = if (container_kind == .@"struct")
const default_value = if (t.default_init and container_kind == .@"struct")
try t.createZeroValueNode(field.qt, field_type, .no_as)
else
null;
@@ -616,7 +713,7 @@ fn transRecordDecl(t: *Translator, scope: *Scope, record_qt: QualType) Error!voi
.name = "_padding",
.type = try ZigTag.type.create(t.arena, try std.fmt.allocPrint(t.arena, "u{d}", .{padding_bits})),
.alignment = @divExact(alignment_bits, 8),
.default_value = if (container_kind == .@"struct")
.default_value = if (t.default_init and container_kind == .@"struct")
ZigTag.zero_literal.init()
else
null,
@@ -663,14 +760,12 @@ fn transRecordDecl(t: *Translator, scope: *Scope, record_qt: QualType) Error!voi
fn transFnDecl(t: *Translator, scope: *Scope, function: Node.Function) Error!void {
const func_ty = function.qt.get(t.comp, .func).?;
const is_pub = scope.id == .root;
const fn_name = t.tree.tokSlice(function.name_tok);
if (scope.getAlias(fn_name) != null or t.global_scope.containsNow(fn_name))
return; // Avoid processing this decl twice
const fn_decl_loc = function.name_tok;
const has_body = function.body != null and func_ty.kind != .variadic;
const has_body = function.body != null and func_ty.kind != .variadic and t.func_bodies;
if (function.body != null and func_ty.kind == .variadic) {
try t.warn(scope, function.name_tok, "TODO unable to translate variadic function, demoted to extern", .{});
}
@@ -681,7 +776,7 @@ fn transFnDecl(t: *Translator, scope: *Scope, function: Node.Function) Error!voi
.is_always_inline = is_always_inline,
.is_extern = !has_body,
.is_export = !function.static and has_body and !is_always_inline and !function.@"inline",
.is_pub = is_pub,
.is_pub = scope.id == .root and (!function.static or t.pub_static),
.has_body = has_body,
.cc = if (function.qt.getAttribute(t.comp, .calling_convention)) |some| switch (some.cc) {
.c => .c,
@@ -761,6 +856,7 @@ fn transFnDecl(t: *Translator, scope: *Scope, function: Node.Function) Error!voi
t.transCompoundStmtInline(body_stmt, &block_scope) catch |err| switch (err) {
error.OutOfMemory => |e| return e,
error.SelfReferential => unreachable,
error.UnsupportedTranslation,
error.UnsupportedType,
=> {
@@ -777,7 +873,7 @@ fn transFnDecl(t: *Translator, scope: *Scope, function: Node.Function) Error!voi
return t.addTopLevelDecl(fn_name, proto_node);
}
fn transVarDecl(t: *Translator, scope: *Scope, variable: Node.Variable) Error!void {
fn transVarDecl(t: *Translator, scope: *Scope, variable: Node.Variable, decl_node: Node.Index) Error!void {
const base_name = t.tree.tokSlice(variable.name_tok);
const toplevel = scope.id == .root;
const bs: *Scope.Block = if (!toplevel) try scope.findBlockScope(t) else undefined;
@@ -815,24 +911,28 @@ fn transVarDecl(t: *Translator, scope: *Scope, variable: Node.Variable) Error!vo
var is_const = variable.qt.@"const" or (array_ty != null and array_ty.?.elem.@"const");
var is_extern = variable.storage_class == .@"extern";
var self_referential = false;
const init_node = init: {
if (variable.initializer) |init| {
const maybe_literal = init.get(t.tree);
if (!toplevel) try t.wip_var_inits.putNoClobber(t.gpa, decl_node, {});
defer _ = t.wip_var_inits.remove(decl_node);
const init_node = (if (maybe_literal == .string_literal_expr)
t.transStringLiteralInitializer(init, maybe_literal.string_literal_expr, type_node)
else
t.transExprCoercing(scope, init, .used)) catch |err| switch (err) {
error.SelfReferential => {
self_referential = true;
break :init ZigTag.undefined_literal.init();
},
error.UnsupportedTranslation, error.UnsupportedType => {
return t.failDecl(scope, variable.name_tok, name, "unable to resolve var init expr", .{});
},
else => |e| return e,
};
if (!variable.qt.is(t.comp, .bool) and init_node.isBoolRes()) {
break :init try ZigTag.int_from_bool.create(t.arena, init_node);
} else {
break :init init_node;
}
break :init try t.toNonBool(init_node, variable.qt);
}
if (variable.storage_class == .@"extern") {
if (array_ty != null and array_ty.?.len == .incomplete) {
@@ -876,7 +976,7 @@ fn transVarDecl(t: *Translator, scope: *Scope, variable: Node.Variable) Error!vo
const alignment: ?c_uint = variable.qt.requestedAlignment(t.comp) orelse null;
var node = try ZigTag.var_decl.create(t.arena, .{
.is_pub = toplevel,
.is_const = is_const,
.is_const = is_const and !self_referential,
.is_extern = is_extern,
.is_export = toplevel and variable.storage_class == .auto and linkage == .strong,
.is_threadlocal = variable.thread_local,
@@ -894,6 +994,21 @@ fn transVarDecl(t: *Translator, scope: *Scope, variable: Node.Variable) Error!vo
node = try ZigTag.wrapped_local.create(t.arena, .{ .name = name, .init = node });
}
try scope.appendNode(node);
if (self_referential) {
const deferred_init = t.transExprCoercing(scope, variable.initializer.?, .used) catch |err| switch (err) {
error.SelfReferential => unreachable,
error.UnsupportedTranslation, error.UnsupportedType => {
return t.failDecl(scope, variable.name_tok, name, "unable to resolve var init expr", .{});
},
else => |e| return e,
};
const assign = try ZigTag.assign.create(t.arena, .{
.lhs = try ZigTag.identifier.create(t.arena, name),
.rhs = try t.toNonBool(deferred_init, variable.qt),
});
try scope.appendNode(assign);
}
try bs.discardVariable(name);
if (variable.qt.getAttribute(t.comp, .cleanup)) |cleanup_attr| {
@@ -1001,6 +1116,7 @@ fn transEnumDecl(t: *Translator, scope: *Scope, enum_qt: QualType) Error!void {
fn transStaticAssert(t: *Translator, scope: *Scope, static_assert: Node.StaticAssert) Error!void {
const condition = t.transExpr(scope, static_assert.cond, .used) catch |err| switch (err) {
error.SelfReferential => unreachable,
error.UnsupportedTranslation, error.UnsupportedType => {
return try t.warn(&t.global_scope.base, static_assert.cond.tok(t.tree), "unable to translate _Static_assert condition", .{});
},
@@ -1084,21 +1200,17 @@ fn transType(t: *Translator, scope: *Scope, qt: QualType, source_loc: TokenIndex
},
.float => |float_ty| switch (float_ty) {
.fp16, .float16 => return ZigTag.type.create(t.arena, "f16"),
.float => return ZigTag.type.create(t.arena, "f32"),
.double => return ZigTag.type.create(t.arena, "f64"),
.long_double => return ZigTag.type.create(t.arena, "c_longdouble"),
.float, .float32 => return ZigTag.type.create(t.arena, "f32"),
.double, .float64, .float32x => return ZigTag.type.create(t.arena, "f64"),
.long_double, .float64x => return ZigTag.type.create(t.arena, "c_longdouble"),
.float128 => return ZigTag.type.create(t.arena, "f128"),
.bf16,
.float32,
.float64,
.float32x,
.float64x,
.float128x,
.bf16 => return t.fail(error.UnsupportedType, source_loc, "TODO support bfloat16", .{}),
.dfloat32,
.dfloat64,
.dfloat128,
.dfloat64x,
=> return t.fail(error.UnsupportedType, source_loc, "TODO support float type: '{s}'", .{try t.getTypeStr(qt)}),
=> return t.fail(error.UnsupportedType, source_loc, "TODO support decimal float type: '{s}'", .{try t.getTypeStr(qt)}),
.float128x => unreachable, // Unsupported on all targets
},
.pointer => |pointer_ty| {
const child_qt = pointer_ty.child;
@@ -1173,9 +1285,21 @@ fn transType(t: *Translator, scope: *Scope, qt: QualType, source_loc: TokenIndex
return ZigTag.identifier.create(t.arena, name);
},
.attributed => |attributed_ty| continue :loop attributed_ty.base.type(t.comp),
.typeof => |typeof_ty| continue :loop typeof_ty.base.type(t.comp),
.typeof => |typeof_ty| {
if (typeof_ty.expr) |expr| {
if (t.transExpr(scope, expr, .used)) |node| {
return ZigTag.typeof.create(t.arena, node);
} else |err| switch (err) {
error.SelfReferential => {},
error.UnsupportedTranslation => {},
error.UnsupportedType => {},
error.OutOfMemory => return error.OutOfMemory,
}
}
continue :loop typeof_ty.base.type(t.comp);
},
.vector => |vector_ty| {
const len = try t.createNumberNode(vector_ty.len, .int);
const len = try t.createNumberNode(vector_ty.len);
const elem_type = try t.transType(scope, vector_ty.elem, source_loc);
return ZigTag.vector.create(t.arena, .{ .lhs = len, .rhs = elem_type });
},
@@ -1384,7 +1508,10 @@ fn transFnType(
.is_var_args = switch (func_ty.kind) {
.normal => false,
.variadic => true,
.old_style => !ctx.is_export and !ctx.is_always_inline and !ctx.has_body,
.old_style => if (t.comp.target.cpu.arch.isWasm())
false
else
!ctx.is_export and !ctx.is_always_inline and !ctx.has_body,
},
.name = ctx.fn_name,
.linksection_string = linksection_string,
@@ -1468,7 +1595,7 @@ fn typeIsOpaque(t: *Translator, qt: QualType) bool {
}
fn typeWasDemotedToOpaque(t: *Translator, qt: QualType) bool {
return t.opaque_demotes.contains(qt);
return t.opaque_demotes.contains(qt.base(t.comp).qt);
}
fn typeHasWrappingOverflow(t: *Translator, qt: QualType) bool {
@@ -1531,16 +1658,30 @@ fn transStmt(t: *Translator, scope: *Scope, stmt: Node.Index) TransError!ZigNode
try t.transRecordDecl(scope, record_decl.container_qt);
return ZigTag.declaration.init();
},
.struct_forward_decl, .union_forward_decl => |record_decl| {
if (record_decl.definition) |some| {
return t.transStmt(scope, some);
}
try t.transRecordDecl(scope, record_decl.container_qt);
return ZigTag.declaration.init();
},
.enum_decl => |enum_decl| {
try t.transEnumDecl(scope, enum_decl.container_qt);
return ZigTag.declaration.init();
},
.enum_forward_decl => |enum_decl| {
if (enum_decl.definition) |some| {
return t.transStmt(scope, some);
}
try t.transEnumDecl(scope, enum_decl.container_qt);
return ZigTag.declaration.init();
},
.function => |function| {
try t.transFnDecl(scope, function);
return ZigTag.declaration.init();
},
.variable => |variable| {
try t.transVarDecl(scope, variable);
try t.transVarDecl(scope, variable, stmt);
return ZigTag.declaration.init();
},
.switch_stmt => |switch_stmt| return t.transSwitch(scope, switch_stmt),
@@ -1562,7 +1703,10 @@ fn transCompoundStmtInline(t: *Translator, compound: Node.CompoundStmt, block: *
const result = try t.transStmt(&block.base, stmt);
switch (result.tag()) {
.declaration, .empty_block => {},
else => try block.statements.append(t.gpa, result),
else => {
try block.statements.append(t.gpa, result);
if (result.isNoreturn()) return;
},
}
}
}
@@ -1578,12 +1722,9 @@ fn transReturnStmt(t: *Translator, scope: *Scope, return_stmt: Node.ReturnStmt)
switch (return_stmt.operand) {
.none => return ZigTag.return_void.init(),
.expr => |operand| {
var rhs = try t.transExprCoercing(scope, operand, .used);
const rhs = try t.transExprCoercing(scope, operand, .used);
const return_qt = scope.findBlockReturnType();
if (rhs.isBoolRes() and !return_qt.is(t.comp, .bool)) {
rhs = try ZigTag.int_from_bool.create(t.arena, rhs);
}
return ZigTag.@"return".create(t.arena, rhs);
return ZigTag.@"return".create(t.arena, try t.toNonBool(rhs, return_qt));
},
.implicit => |zero| {
if (zero) return ZigTag.@"return".create(t.arena, ZigTag.zero_literal.init());
@@ -1698,7 +1839,7 @@ fn transDoWhileStmt(t: *Translator, scope: *Scope, do_stmt: Node.DoWhileStmt) Tr
};
var body_node = try t.transStmt(&loop_scope, do_stmt.body);
if (body_node.isNoreturn(true)) {
if (body_node.isNoreturn()) {
// The body node ends in a noreturn statement. Simply put it in a while (true)
// in case it contains breaks or continues.
} else if (do_stmt.body.get(t.tree) == .compound_stmt) {
@@ -1918,8 +2059,6 @@ fn transSwitchProngStmt(
body: []const Node.Index,
) TransError!ZigNode {
switch (stmt.get(t.tree)) {
.break_stmt => return ZigTag.@"break".init(),
.return_stmt => return t.transStmt(scope, stmt),
.case_stmt, .default_stmt => unreachable,
else => {
var block_scope = try Scope.Block.init(t, scope, false);
@@ -1940,15 +2079,6 @@ fn transSwitchProngStmtInline(
) TransError!void {
for (body) |stmt| {
switch (stmt.get(t.tree)) {
.return_stmt => {
const result = try t.transStmt(&block.base, stmt);
try block.statements.append(t.gpa, result);
return;
},
.break_stmt => {
try block.statements.append(t.gpa, ZigTag.@"break".init());
return;
},
.case_stmt => |case_stmt| {
var sub = case_stmt.body;
while (true) switch (sub.get(t.tree)) {
@@ -1959,7 +2089,7 @@ fn transSwitchProngStmtInline(
const result = try t.transStmt(&block.base, sub);
assert(result.tag() != .declaration);
try block.statements.append(t.gpa, result);
if (result.isNoreturn(true)) return;
if (result.isNoreturn()) return;
},
.default_stmt => |default_stmt| {
var sub = default_stmt.body;
@@ -1971,18 +2101,16 @@ fn transSwitchProngStmtInline(
const result = try t.transStmt(&block.base, sub);
assert(result.tag() != .declaration);
try block.statements.append(t.gpa, result);
if (result.isNoreturn(true)) return;
},
.compound_stmt => |compound_stmt| {
const result = try t.transCompoundStmt(&block.base, compound_stmt);
try block.statements.append(t.gpa, result);
if (result.isNoreturn(true)) return;
if (result.isNoreturn()) return;
},
else => {
const result = try t.transStmt(&block.base, stmt);
switch (result.tag()) {
.declaration, .empty_block => {},
else => try block.statements.append(t.gpa, result),
else => {
try block.statements.append(t.gpa, result);
if (result.isNoreturn()) return;
},
}
},
}
@@ -2015,7 +2143,14 @@ fn transExpr(t: *Translator, scope: *Scope, expr: Node.Index, used: ResultUsed)
break :res try ZigTag.deref.create(t.arena, try t.transExpr(scope, deref_expr.operand, .used));
},
.bool_not_expr => |bool_not_expr| try ZigTag.not.create(t.arena, try t.transBoolExpr(scope, bool_not_expr.operand)),
.bit_not_expr => |bit_not_expr| try ZigTag.bit_not.create(t.arena, try t.transExpr(scope, bit_not_expr.operand, .used)),
.bit_not_expr => |bit_not_expr| try ZigTag.bit_not.create(t.arena, op: {
const operand = try t.transExpr(scope, bit_not_expr.operand, .used);
if (!operand.isBoolRes()) break :op operand;
const casted = try ZigTag.int_from_bool.create(t.arena, operand);
const ty = try t.transType(scope, bit_not_expr.qt, bit_not_expr.op_tok);
break :op try ZigTag.as.create(t.arena, .{ .lhs = ty, .rhs = casted });
}),
.plus_expr => |plus_expr| return t.transExpr(scope, plus_expr.operand, used),
.negate_expr => |negate_expr| res: {
const operand_qt = negate_expr.operand.qt(t.tree);
@@ -2109,8 +2244,8 @@ fn transExpr(t: *Translator, scope: *Scope, expr: Node.Index, used: ResultUsed)
.shl_expr => |shl_expr| try t.transShiftExpr(scope, shl_expr, .shl),
.shr_expr => |shr_expr| try t.transShiftExpr(scope, shr_expr, .shr),
.member_access_expr => |member_access| try t.transMemberAccess(scope, .normal, member_access, null),
.member_access_ptr_expr => |member_access| try t.transMemberAccess(scope, .ptr, member_access, null),
.member_access_expr => |member_access| try t.transMemberAccess(scope, .normal, member_access, null, .accessor),
.member_access_ptr_expr => |member_access| try t.transMemberAccess(scope, .ptr, member_access, null, .accessor),
.array_access_expr => |array_access| try t.transArrayAccess(scope, array_access, null),
.builtin_ref => unreachable,
@@ -2195,6 +2330,10 @@ fn transExpr(t: *Translator, scope: *Scope, expr: Node.Index, used: ResultUsed)
.builtin_convertvector => |convertvector| try t.transConvertvectorExpr(scope, convertvector),
.builtin_shufflevector => |shufflevector| try t.transShufflevectorExpr(scope, shufflevector),
.builtin_va_arg_pack, .builtin_va_arg_pack_len => |va_arg_pack| {
return t.fail(error.UnsupportedTranslation, va_arg_pack.builtin_tok, "TODO va arg pack", .{});
},
.compound_stmt,
.static_assert,
.return_stmt,
@@ -2293,6 +2432,12 @@ fn transBoolExpr(t: *Translator, scope: *Scope, expr: Node.Index) TransError!Zig
return t.finishBoolExpr(expr.qt(t.tree), maybe_bool_res);
}
fn toNonBool(t: *Translator, node: ZigNode, qt: QualType) Error!ZigNode {
if (!node.isBoolRes()) return node;
if (qt.is(t.comp, .bool)) return node;
return ZigTag.int_from_bool.create(t.arena, node);
}
fn finishBoolExpr(t: *Translator, qt: QualType, node: ZigNode) TransError!ZigNode {
const sk = qt.scalarKind(t.comp);
if (sk == .bool) return node;
@@ -2385,8 +2530,22 @@ fn transCastExpr(
else => {},
}
if (cast.operand.qt(t.tree).arrayLen(t.comp) == null) {
return try t.transExpr(scope, cast.operand, used);
// Flexible array members are translated as member functions returning
// [*c]T, so no address-of + @ptrCast wrapping is needed.
flexible: {
if (cast.operand.qt(t.tree).arrayLen(t.comp) == null) {
return try t.transExpr(scope, cast.operand, used);
}
const member_index, const base_qt = switch (cast.operand.get(t.tree)) {
.member_access_expr => |ma| .{ ma.member_index, ma.base.qt(t.tree) },
.member_access_ptr_expr => |ma| .{ ma.member_index, ma.base.qt(t.tree).childType(t.comp) },
else => break :flexible,
};
const record = base_qt.getRecord(t.comp) orelse break :flexible;
if (member_index != record.fields.len - 1 and base_qt.base(t.comp).type != .@"union") break :flexible;
const array_ty = record.fields[member_index].qt.get(t.comp, .array) orelse break :flexible;
if (t.isFlexibleArrayLen(array_ty.len)) return try t.transExpr(scope, cast.operand, used);
}
const sub_expr_node = try t.transExpr(scope, cast.operand, .used);
@@ -2402,6 +2561,8 @@ fn transCastExpr(
.lhs = try ZigTag.type.create(t.arena, "usize"),
.rhs = try ZigTag.int_cast.create(t.arena, sub_expr_node),
});
} else if (sub_expr_node.isBoolRes()) {
sub_expr_node = try ZigTag.int_from_bool.create(t.arena, sub_expr_node);
}
break :int_to_pointer try ZigTag.ptr_from_int.create(t.arena, sub_expr_node);
},
@@ -2560,6 +2721,8 @@ fn transPointerCastExpr(t: *Translator, scope: *Scope, expr: Node.Index) TransEr
}
fn transDeclRefExpr(t: *Translator, scope: *Scope, decl_ref: Node.DeclRef) TransError!ZigNode {
if (t.wip_var_inits.contains(decl_ref.decl)) return error.SelfReferential;
const name = t.tree.tokSlice(decl_ref.name_tok);
const maybe_alias = scope.getAlias(name);
const mangled_name = maybe_alias orelse name;
@@ -2631,7 +2794,7 @@ fn transShiftExpr(t: *Translator, scope: *Scope, bin: Node.Binary, op_id: ZigTag
// lhs >> @intCast(rh)
const lhs = try t.transExpr(scope, bin.lhs, .used);
const rhs = try t.transExprCoercing(scope, bin.rhs, .used);
const rhs = try t.transExpr(scope, bin.rhs, .used);
const rhs_casted = try ZigTag.int_cast.create(t.arena, rhs);
return t.createBinOpNode(op_id, lhs, rhs_casted);
@@ -2758,7 +2921,7 @@ fn transCommaExpr(t: *Translator, scope: *Scope, bin: Node.Binary, used: ResultU
const rhs = try t.transExprCoercing(&block_scope.base, bin.rhs, .used);
const break_node = try ZigTag.break_val.create(t.arena, .{
.label = block_scope.label,
.val = rhs,
.val = try t.toNonBool(rhs, bin.qt),
});
try block_scope.statements.append(t.gpa, break_node);
@@ -2768,14 +2931,10 @@ fn transCommaExpr(t: *Translator, scope: *Scope, bin: Node.Binary, used: ResultU
fn transAssignExpr(t: *Translator, scope: *Scope, bin: Node.Binary, used: ResultUsed) !ZigNode {
if (used == .unused) {
const lhs = try t.transExpr(scope, bin.lhs, .used);
var rhs = try t.transExprCoercing(scope, bin.rhs, .used);
const rhs = try t.transExprCoercing(scope, bin.rhs, .used);
const lhs_qt = bin.lhs.qt(t.tree);
if (rhs.isBoolRes() and !lhs_qt.is(t.comp, .bool)) {
rhs = try ZigTag.int_from_bool.create(t.arena, rhs);
}
return t.createBinOpNode(.assign, lhs, rhs);
return t.createBinOpNode(.assign, lhs, try t.toNonBool(rhs, lhs_qt));
}
var block_scope = try Scope.Block.init(t, scope, true);
@@ -2783,13 +2942,12 @@ fn transAssignExpr(t: *Translator, scope: *Scope, bin: Node.Binary, used: Result
const tmp = try block_scope.reserveMangledName("tmp");
var rhs = try t.transExpr(&block_scope.base, bin.rhs, .used);
const rhs = try t.transExpr(&block_scope.base, bin.rhs, .used);
const lhs_qt = bin.lhs.qt(t.tree);
if (rhs.isBoolRes() and !lhs_qt.is(t.comp, .bool)) {
rhs = try ZigTag.int_from_bool.create(t.arena, rhs);
}
const tmp_decl = try ZigTag.var_simple.create(t.arena, .{ .name = tmp, .init = rhs });
const tmp_decl = try ZigTag.var_simple.create(t.arena, .{
.name = tmp,
.init = try t.toNonBool(rhs, lhs_qt),
});
try block_scope.statements.append(t.gpa, tmp_decl);
const lhs = try t.transExprCoercing(&block_scope.base, bin.lhs, .used);
@@ -3040,6 +3198,7 @@ fn transMemberAccess(
kind: enum { normal, ptr },
member_access: Node.MemberAccess,
opt_base: ?ZigNode,
flex_array_mode: enum { accessor, backing },
) TransError!ZigNode {
const base_info = switch (kind) {
.normal => member_access.base.qt(t.tree),
@@ -3068,8 +3227,14 @@ fn transMemberAccess(
// Flexible array members are translated as member functions.
if (member_access.member_index == record.fields.len - 1 or base_info.base(t.comp).type == .@"union") {
if (field.qt.get(t.comp, .array)) |array_ty| {
if (array_ty.len == .incomplete or (array_ty.len == .fixed and array_ty.len.fixed == 0)) {
return ZigTag.call.create(t.arena, .{ .lhs = field_access, .args = &.{} });
if (t.isFlexibleArrayLen(array_ty.len)) {
switch (flex_array_mode) {
.accessor => return ZigTag.call.create(t.arena, .{ .lhs = field_access, .args = &.{} }),
.backing => {
const backing_name = try std.fmt.allocPrint(t.arena, "_{s}", .{field_name});
return ZigTag.field_access.create(t.arena, .{ .lhs = lhs, .field_name = backing_name });
},
}
}
}
}
@@ -3091,7 +3256,7 @@ fn transArrayAccess(t: *Translator, scope: *Scope, array_access: Node.ArrayAcces
const index = index: {
const index = try t.transExpr(scope, array_access.index, .used);
const index_qt = array_access.index.qt(t.tree);
const maybe_bigger_than_usize = switch (index_qt.base(t.comp).type) {
const maybe_bigger_than_usize = type: switch (index_qt.base(t.comp).type) {
.bool => {
break :index try ZigTag.int_from_bool.create(t.arena, index);
},
@@ -3100,6 +3265,7 @@ fn transArrayAccess(t: *Translator, scope: *Scope, array_access: Node.ArrayAcces
else => false,
},
.bit_int => |bit_int| bit_int.bits > t.comp.target.ptrBitWidth(),
.@"enum" => |e| if (e.tag) |tag| continue :type tag.base(t.comp).type else false,
else => unreachable,
};
@@ -3158,7 +3324,10 @@ fn transMemberDesignator(t: *Translator, scope: *Scope, arg: Node.Index) TransEr
},
.member_access_expr => |access| {
const base = try t.transMemberDesignator(scope, access.base);
return t.transMemberAccess(scope, .normal, access, base);
// In offsetof context, flexible array members must be accessed via
// the backing field (`_name`) rather than the accessor function,
// because you can't take the address of a function call result.
return t.transMemberAccess(scope, .normal, access, base, .backing);
},
.cast => |cast| {
assert(cast.kind == .array_to_pointer);
@@ -3292,6 +3461,51 @@ fn transCall(
const SuppressCast = enum { with_as, no_as };
/// Attempt to translate literal as the name of the simple macro
/// it was expanded from.
fn checkLiteralMacro(t: *Translator, tok: TokenIndex, used: ResultUsed) !?ZigNode {
if (!t.keep_macro_literals) return null;
const expansion_locs = t.pp.expansionSlice(tok);
if (expansion_locs.len == 0) return null;
const last_expand = expansion_locs[0];
const source = t.comp.getSource(last_expand.id);
var tokenizer: aro.Tokenizer = .{
.buf = source.buf,
.langopts = t.comp.langopts,
.source = last_expand.id,
.index = last_expand.byte_offset,
.splice_locs = &.{},
};
const name_tok = tokenizer.next();
if (!name_tok.id.isMacroIdentifier()) return null;
const name = t.pp.tokSlice(name_tok);
if (t.global_scope.containsNow(name)) return null;
const macro = t.pp.defines.get(name) orelse return null;
if (macro.is_func) return null;
if (macro.isBuiltin()) return null;
var tok_count: u8 = 0;
for (macro.tokens) |macro_tok| {
switch (macro_tok.id) {
.invalid => continue,
.whitespace => continue,
.comment => continue,
.macro_ws => continue,
else => {
if (tok_count != 0) return null;
tok_count += 1;
},
}
}
if (t.checkTranslatableMacro(macro.tokens, macro.params) != null) return null;
const ident = try ZigTag.identifier.create(t.arena, name);
return try t.maybeSuppressResult(used, ident);
}
fn transIntLiteral(
t: *Translator,
scope: *Scope,
@@ -3299,6 +3513,7 @@ fn transIntLiteral(
used: ResultUsed,
suppress_as: SuppressCast,
) TransError!ZigNode {
if (try t.checkLiteralMacro(literal_index.tok(t.tree), used)) |node| return node;
const val = t.tree.value_map.get(literal_index).?;
const int_lit_node = try t.createIntNode(val);
if (suppress_as == .no_as) {
@@ -3325,6 +3540,7 @@ fn transCharLiteral(
used: ResultUsed,
suppress_as: SuppressCast,
) TransError!ZigNode {
if (try t.checkLiteralMacro(literal_index.tok(t.tree), used)) |node| return node;
const val = t.tree.value_map.get(literal_index).?;
const char_literal = literal_index.get(t.tree).char_literal;
const narrow = char_literal.kind == .ascii or char_literal.kind == .utf8;
@@ -3333,7 +3549,7 @@ fn transCharLiteral(
// e.g. 'abcd'
const int_value = val.toInt(u32, t.comp).?;
const int_lit_node = if (char_literal.kind == .ascii and int_value > 255)
try t.createNumberNode(int_value, .int)
try t.createNumberNode(int_value)
else
try t.createCharLiteralNode(narrow, int_value);
@@ -3357,12 +3573,16 @@ fn transFloatLiteral(
used: ResultUsed,
suppress_as: SuppressCast,
) TransError!ZigNode {
if (try t.checkLiteralMacro(literal_index.tok(t.tree), used)) |node| return node;
const val = t.tree.value_map.get(literal_index).?;
const float_literal = literal_index.get(t.tree).float_literal;
var allocating: std.Io.Writer.Allocating = .init(t.gpa);
defer allocating.deinit();
_ = val.print(float_literal.qt, t.comp, &allocating.writer) catch return error.OutOfMemory;
if (mem.findScalar(u8, allocating.written(), '.') == null) {
allocating.writer.writeAll(".0") catch return error.OutOfMemory;
}
const float_lit_node = try ZigTag.float_literal.create(t.arena, try t.arena.dupe(u8, allocating.written()));
if (suppress_as == .no_as) {
@@ -3588,7 +3808,7 @@ fn transArrayInit(
while (i < array_init.items.len) : (i += 1) {
if (array_init.items[i].get(t.tree) == .array_filler_expr) break;
const expr = try t.transExprCoercing(scope, array_init.items[i], .used);
try val_list.append(t.gpa, expr);
try val_list.append(t.gpa, try t.toNonBool(expr, array_item_qt));
}
const array_type = try ZigTag.array_type.create(t.arena, .{
.elem_type = array_item_type,
@@ -3638,7 +3858,7 @@ fn transUnionInit(
const field_init = try t.arena.create(ast.Payload.ContainerInit.Initializer);
field_init.* = .{
.name = field_name,
.value = try t.transExprCoercing(scope, init_expr, .used),
.value = try t.toNonBool(try t.transExprCoercing(scope, init_expr, .used), field.qt),
};
const container_init = try ZigTag.container_init.create(t.arena, .{
.lhs = union_type,
@@ -3669,7 +3889,7 @@ fn transStructInit(
}).? else field.name.lookup(t.comp);
init.* = .{
.name = field_name,
.value = try t.transExprCoercing(scope, field_expr, .used),
.value = try t.toNonBool(try t.transExprCoercing(scope, field_expr, .used), field.qt),
};
}
@@ -3766,7 +3986,7 @@ fn transConvertvectorExpr(
for (items, 0..dest_vec_ty.len) |*item, i| {
const value = try ZigTag.array_access.create(t.arena, .{
.lhs = tmp_ident,
.rhs = try t.createNumberNode(i, .int),
.rhs = try t.createNumberNode(i),
});
if (src_elem_sk == .float and dest_elem_sk == .float) {
@@ -3812,7 +4032,7 @@ fn transShufflevectorExpr(
const mask_len = shufflevector.indexes.len;
const mask_type = try ZigTag.vector.create(t.arena, .{
.lhs = try t.createNumberNode(mask_len, .int),
.lhs = try t.createNumberNode(mask_len),
.rhs = try ZigTag.type.create(t.arena, "i32"),
});
@@ -3882,16 +4102,9 @@ fn createIntNode(t: *Translator, int: aro.Value) !ZigNode {
return res;
}
fn createNumberNode(t: *Translator, num: anytype, num_kind: enum { int, float }) !ZigNode {
const fmt_s = switch (@typeInfo(@TypeOf(num))) {
.int, .comptime_int => "{d}",
else => "{s}",
};
const str = try std.fmt.allocPrint(t.arena, fmt_s, .{num});
if (num_kind == .float)
return ZigTag.float_literal.create(t.arena, str)
else
return ZigTag.integer_literal.create(t.arena, str);
fn createNumberNode(t: *Translator, num: anytype) !ZigNode {
const str = try std.fmt.allocPrint(t.arena, "{d}", .{num});
return ZigTag.integer_literal.create(t.arena, str);
}
fn createCharLiteralNode(t: *Translator, narrow: bool, val: u32) TransError!ZigNode {
@@ -3953,6 +4166,25 @@ fn vectorTypeInfo(t: *Translator, vec_node: ZigNode, field: []const u8) TransErr
return ZigTag.field_access.create(t.arena, .{ .lhs = vector_type_info, .field_name = field });
}
/// Returns true if the given array length qualifies as a flexible array member
/// under the current -fstrict-flex-arrays level.
fn isFlexibleArrayLen(t: *const Translator, len: anytype) bool {
return switch (t.strict_flex_arrays) {
.@"0" => true,
.@"1" => switch (len) {
.incomplete => true,
.fixed => |n| n <= 1,
else => false,
},
.@"2" => switch (len) {
.incomplete => true,
.fixed => |n| n == 0,
else => false,
},
.@"3" => len == .incomplete,
};
}
/// Build a getter function for a flexible array field in a C record
/// e.g. `T items[]` or `T items[0]`. The generated function returns a [*c] pointer
/// to the flexible array with the correct const and volatile qualifiers
@@ -3961,7 +4193,13 @@ fn createFlexibleMemberFn(
member_name: []const u8,
field_name: []const u8,
) Error!ZigNode {
const self_param_name = "self";
// Use `_self` instead of the conventional `self` to avoid the Zig error
// "function parameter shadows declaration of 'self'".
// `processContainerMemberFns` merges C functions matching a struct's name
// prefix into the struct as `pub const` aliases (e.g. `foo_self()` becomes
// `pub const self = __root.foo_self`). A parameter also named `self` would
// then shadow that declaration, which Zig rejects.
const self_param_name = "_self";
const self_param = try ZigTag.identifier.create(t.arena, self_param_name);
const self_type = try ZigTag.typeof.create(t.arena, self_param);
+19 -11
View File
@@ -50,6 +50,7 @@ pub const Node = extern union {
break_val,
@"return",
field_access,
field_builtin,
array_access,
call,
var_decl,
@@ -371,6 +372,7 @@ pub const Node = extern union {
.div_exact,
.offset_of,
.static_assert,
.field_builtin,
=> Payload.BinOp,
.integer_literal,
@@ -455,14 +457,14 @@ pub const Node = extern union {
return .{ .ptr_otherwise = payload };
}
pub fn isNoreturn(node: Node, break_counts: bool) bool {
switch (node.tag()) {
pub fn isNoreturn(node: Node) bool {
return switch (node.tag()) {
.block => {
const block_node = node.castTag(.block).?;
if (block_node.data.stmts.len == 0) return false;
const last = block_node.data.stmts[block_node.data.stmts.len - 1];
return last.isNoreturn(break_counts);
return last.isNoreturn();
},
.@"switch" => {
const switch_node = node.castTag(.@"switch").?;
@@ -475,15 +477,16 @@ pub const Node = extern union {
else
unreachable;
if (!body.isNoreturn(break_counts)) return false;
if (!body.isNoreturn()) return false;
}
return true;
},
.@"return", .return_void => return true,
.@"break" => if (break_counts) return true,
else => {},
}
return false;
.@"return", .return_void => true,
.@"break" => true,
.@"continue" => true,
.@"unreachable" => true,
else => false,
};
}
pub fn isBoolRes(res: Node) bool {
@@ -2015,6 +2018,10 @@ fn renderNode(c: *Context, node: Node) Allocator.Error!NodeIndex {
const lhs = try renderNodeGrouped(c, payload.lhs);
return renderFieldAccess(c, lhs, payload.field_name);
},
.field_builtin => {
const payload = node.castTag(.field_builtin).?.data;
return renderBuiltinCall(c, "@field", &.{ payload.lhs, payload.rhs });
},
.@"struct", .@"union", .@"opaque" => return renderContainer(c, node),
.enum_constant => {
const payload = node.castTag(.enum_constant).?.data;
@@ -2424,7 +2431,7 @@ fn renderNullSentinelArrayType(c: *Context, len: u64, elem_type: Node) !NodeInde
fn addSemicolonIfNeeded(c: *Context, node: Node) !void {
switch (node.tag()) {
.warning => unreachable,
.var_decl, .var_simple, .arg_redecl, .alias, .block, .empty_block, .block_single, .@"switch", .wrapped_local, .mut_str => {},
.static_assert, .var_decl, .var_simple, .arg_redecl, .alias, .block, .empty_block, .block_single, .@"switch", .wrapped_local, .mut_str => {},
.while_true => {
const payload = node.castTag(.while_true).?.data;
return addSemicolonIfNotBlock(c, payload);
@@ -2532,6 +2539,8 @@ fn renderNodeGrouped(c: *Context, node: Node) !NodeIndex {
.trunc,
.floor,
.root_ref,
.field_builtin,
.@"switch",
=> {
// no grouping needed
return renderNode(c, node);
@@ -2594,7 +2603,6 @@ fn renderNodeGrouped(c: *Context, node: Node) !NodeIndex {
.pub_var_simple,
.enum_constant,
.@"while",
.@"switch",
.@"break",
.break_val,
.pub_inline_fn,
+75 -16
View File
@@ -1,15 +1,17 @@
const std = @import("std");
const Io = std.Io;
const assert = std.debug.assert;
const mem = std.mem;
const process = std.process;
const Io = std.Io;
const aro = @import("aro");
const compiler_util = @import("../util.zig");
const Translator = @import("Translator.zig");
const fast_exit = @import("builtin").mode != .Debug;
pub fn main(init: std.process.Init) u8 {
pub fn main(init: process.Init) u8 {
const gpa = init.gpa;
const arena = init.arena.allocator();
const io = init.io;
@@ -33,16 +35,20 @@ pub fn main(init: std.process.Init) u8 {
var stderr = Io.File.stderr().writer(io, &stderr_buf);
var diagnostics: aro.Diagnostics = switch (zig_integration) {
false => .{ .output = .{ .to_writer = .{
.mode = Io.Terminal.Mode.detect(io, stderr.file, NO_COLOR, CLICOLOR_FORCE) catch unreachable,
.mode = Io.Terminal.Mode.detect(io, stderr.file, NO_COLOR, CLICOLOR_FORCE) catch .no_color,
.writer = &stderr.interface,
} } },
true => .{ .output = .{ .to_list = .{
.arena = .init(gpa),
} } },
true => .{ .output = .{ .to_list = .{ .arena = .init(gpa) } } },
};
defer diagnostics.deinit();
var comp = aro.Compilation.initDefault(gpa, arena, io, &diagnostics, .cwd(), environ_map) catch |err| switch (err) {
var comp = aro.Compilation.init(.{
.gpa = gpa,
.arena = arena,
.io = io,
.diagnostics = &diagnostics,
.environ_map = environ_map,
}) catch |err| switch (err) {
error.OutOfMemory => {
std.debug.print("ran out of memory initializing C compilation\n", .{});
if (fast_exit) process.exit(1);
@@ -82,7 +88,6 @@ pub fn main(init: std.process.Init) u8 {
return 1;
},
};
assert(comp.diagnostics.errors == 0 or !zig_integration);
if (fast_exit) process.exit(@intFromBool(comp.diagnostics.errors != 0));
return @intFromBool(comp.diagnostics.errors != 0);
@@ -107,10 +112,23 @@ pub const usage =
\\Usage {s}: [options] file [CC options]
\\
\\Options:
\\ --help Print this message
\\ --version Print translate-c version
\\ -fmodule-libs Import libraries as modules
\\ -fno-module-libs (default) Install libraries next to output file
\\ --help Print this message
\\ --version Print translate-c version
\\ -fmodule-libs Import libraries as modules
\\ -fno-module-libs (default) Install libraries next to output file
\\ -fpub-static (default) Translate static functions as pub
\\ -fno-pub-static Do not translate static functions as pub
\\ -ffunc-bodies (default) Translate function bodies
\\ -fno-func-bodies Do not translate function bodies
\\ -fkeep-macro-literals (default) Preserve macro names for literals
\\ -fno-keep-macro-literals Do not preserve macro names for literals
\\ -fdefault-init Default initialize struct fields
\\ -fno-default-init (default) Do not default initialize struct fields
\\ -fstrict-flex-arrays=<n> Control when to treat a trailing array as a flexible array member (default: 2)
\\ 0: any trailing array
\\ 1: size [0]/[1]/[]
\\ 2: size [0]/[]
\\ 3: [] only
\\
\\
;
@@ -119,7 +137,14 @@ fn translate(d: *aro.Driver, tc: *aro.Toolchain, args: []const [:0]const u8, zig
const gpa = d.comp.gpa;
const io = d.comp.io;
var aro_args: std.ArrayList([:0]const u8) = .empty;
var module_libs = true;
var pub_static = true;
var func_bodies = true;
var keep_macro_literals = true;
var default_init = true;
var strict_flex_arrays: Translator.StrictFlexArraysLevel = .@"2";
var aro_args: std.ArrayList([:0]const u8) = try .initCapacity(gpa, args.len);
defer aro_args.deinit(gpa);
for (args, 0..) |arg, i| {
@@ -139,8 +164,34 @@ fn translate(d: *aro.Driver, tc: *aro.Toolchain, args: []const [:0]const u8, zig
} else if (mem.eql(u8, arg, "--zig-integration")) {
if (i != 1 or !zig_integration)
return d.fatal("--zig-integration must be the first argument", .{});
} else if (mem.eql(u8, arg, "-fmodule-libs")) {
module_libs = true;
} else if (mem.eql(u8, arg, "-fno-module-libs")) {
module_libs = false;
} else if (mem.eql(u8, arg, "-fpub-static")) {
pub_static = true;
} else if (mem.eql(u8, arg, "-fno-pub-static")) {
pub_static = false;
} else if (mem.eql(u8, arg, "-ffunc-bodies")) {
func_bodies = true;
} else if (mem.eql(u8, arg, "-fno-func-bodies")) {
func_bodies = false;
} else if (mem.eql(u8, arg, "-fkeep-macro-literals")) {
keep_macro_literals = true;
} else if (mem.eql(u8, arg, "-fno-keep-macro-literals")) {
keep_macro_literals = false;
} else if (mem.eql(u8, arg, "-fdefault-init")) {
default_init = true;
} else if (mem.eql(u8, arg, "-fno-default-init")) {
default_init = false;
} else if (mem.startsWith(u8, arg, "-fstrict-flex-arrays=")) {
const val_str = arg["-fstrict-flex-arrays=".len..];
if (val_str.len != 1 or val_str[0] < '0' or val_str[0] > '3') {
return d.fatal("-fstrict-flex-arrays= requires a value of '0', '1', '2', or '3'", .{});
}
strict_flex_arrays = @enumFromInt(val_str[0] - '0');
} else {
try aro_args.append(gpa, arg);
aro_args.appendAssumeCapacity(arg);
}
}
const user_macros = macros: {
@@ -148,7 +199,7 @@ fn translate(d: *aro.Driver, tc: *aro.Toolchain, args: []const [:0]const u8, zig
defer macro_buf.deinit(gpa);
var discard_buf: [256]u8 = undefined;
var discarding: std.Io.Writer.Discarding = .init(&discard_buf);
var discarding: Io.Writer.Discarding = .init(&discard_buf);
assert(!try d.parseArgs(&discarding.writer, &macro_buf, aro_args.items));
if (macro_buf.items.len > std.math.maxInt(u32)) {
return d.fatal("user provided macro source exceeded max size", .{});
@@ -185,7 +236,9 @@ fn translate(d: *aro.Driver, tc: *aro.Toolchain, args: []const [:0]const u8, zig
else => |e| return e,
};
var pp = try aro.Preprocessor.initDefault(d.comp);
var pp = try aro.Preprocessor.init(d.comp, .{
.base_file = source.id,
});
defer pp.deinit();
var name_buf: [std.fs.max_name_bytes]u8 = undefined;
@@ -235,6 +288,12 @@ fn translate(d: *aro.Driver, tc: *aro.Toolchain, args: []const [:0]const u8, zig
.comp = d.comp,
.pp = &pp,
.tree = &c_tree,
.module_libs = module_libs,
.pub_static = pub_static,
.func_bodies = func_bodies,
.keep_macro_literals = keep_macro_literals,
.default_init = default_init,
.strict_flex_arrays = strict_flex_arrays,
});
defer gpa.free(rendered_zig);
+14 -9
View File
@@ -81,15 +81,20 @@ fn ToUnsigned(comptime T: type) type {
}
/// Constructs a [*c] pointer with the const and volatile annotations
/// from Self for pointing to a C flexible array of Element.
pub fn FlexibleArrayType(comptime Self: type, comptime Element: type) type {
return switch (@typeInfo(Self)) {
.pointer => |ptr| @Pointer(.c, .{
.@"const" = ptr.is_const,
.@"volatile" = ptr.is_volatile,
}, Element, null),
else => |info| @compileError("Invalid self type \"" ++ @tagName(info) ++ "\" for flexible array getter: " ++ @typeName(Self)),
};
/// from SelfType for pointing to a C flexible array of ElementType.
pub fn FlexibleArrayType(comptime SelfType: type, comptime ElementType: type) type {
switch (@typeInfo(SelfType)) {
.pointer => |ptr| {
return @Pointer(.c, .{
.@"const" = ptr.is_const,
.@"volatile" = ptr.is_volatile,
.@"allowzero" = true,
.@"addrspace" = .generic,
.@"align" = null,
}, ElementType, null);
},
else => |info| @compileError("Invalid self type \"" ++ @tagName(info) ++ "\" for flexible array getter: " ++ @typeName(SelfType)),
}
}
/// Promote the type of an integer literal until it fits as C would.