Commit e2974759dd

Veikka Tuominen <git@vexu.eu>
2021-02-17 13:11:49
translate-c: demote untranslatable declarations to externs
1 parent 9a826cc
Changed files (3)
src/translate_c/ast.zig
@@ -1866,6 +1866,12 @@ fn addSemicolonIfNeeded(c: *Context, node: Node) !void {
 fn addSemicolonIfNotBlock(c: *Context, node: Node) !void {
     switch (node.tag()) {
         .block, .empty_block, .block_single, => {},
+        .@"if" => {
+            const payload = node.castTag(.@"if").?.data;
+            if (payload.@"else") |some|
+                return addSemicolonIfNotBlock(c, some);
+            return addSemicolonIfNotBlock(c, payload.then);
+        },
         else => _ = try c.addToken(.semicolon, ";"),
     }
 }
src/translate_c.zig
@@ -508,7 +508,7 @@ fn visitFnDecl(c: *Context, fn_decl: *const clang.FunctionDecl) Error!void {
                 decl_ctx.has_body = false;
                 decl_ctx.storage_class = .Extern;
                 decl_ctx.is_export = false;
-                try warn(c, &c.global_scope.base, fn_decl_loc, "TODO unable to translate variadic function, demoted to declaration", .{});
+                try warn(c, &c.global_scope.base, fn_decl_loc, "TODO unable to translate variadic function, demoted to extern", .{});
             }
             break :blk transFnProto(c, fn_decl, fn_proto_type, fn_decl_loc, decl_ctx, true) catch |err| switch (err) {
                 error.UnsupportedType => {
@@ -543,8 +543,12 @@ fn visitFnDecl(c: *Context, fn_decl: *const clang.FunctionDecl) Error!void {
 
     var param_id: c_uint = 0;
     for (proto_node.data.params) |*param, i| {
-        const param_name = param.name orelse
-            return failDecl(c, fn_decl_loc, fn_name, "function {s} parameter has no name", .{fn_name});
+        const param_name = param.name orelse {
+            proto_node.data.is_extern = true;
+            proto_node.data.is_export = false;
+            try warn(c, &c.global_scope.base, fn_decl_loc, "function {s} parameter has no name, demoted to extern", .{fn_name});
+            return addTopLevelDecl(c, fn_name, Node.initPayload(&proto_node.base));
+        };
 
         const c_param = fn_decl.getParamDecl(param_id);
         const qual_type = c_param.getOriginalType();
@@ -570,7 +574,12 @@ fn visitFnDecl(c: *Context, fn_decl: *const clang.FunctionDecl) Error!void {
         error.OutOfMemory => |e| return e,
         error.UnsupportedTranslation,
         error.UnsupportedType,
-        => return failDecl(c, fn_decl_loc, fn_name, "unable to translate function", .{}),
+        => {
+            proto_node.data.is_extern = true;
+            proto_node.data.is_export = false;
+            try warn(c, &c.global_scope.base, fn_decl_loc, "unable to translate function, demoted to extern", .{});
+            return addTopLevelDecl(c, fn_name, Node.initPayload(&proto_node.base));
+        },
     };
     // add return statement if the function didn't have one
     blk: {
@@ -598,7 +607,12 @@ fn visitFnDecl(c: *Context, fn_decl: *const clang.FunctionDecl) Error!void {
             error.OutOfMemory => |e| return e,
             error.UnsupportedTranslation,
             error.UnsupportedType,
-            => return failDecl(c, fn_decl_loc, fn_name, "unable to create a return value for function", .{}),
+            => {
+                proto_node.data.is_extern = true;
+                proto_node.data.is_export = false;
+                try warn(c, &c.global_scope.base, fn_decl_loc, "unable to create a return value for function, demoted to extern", .{});
+                return addTopLevelDecl(c, fn_name, Node.initPayload(&proto_node.base));
+            },
         };
         const ret = try Tag.@"return".create(c.arena, rhs);
         try block_scope.statements.append(ret);
@@ -641,8 +655,8 @@ fn visitVarDecl(c: *Context, var_decl: *const clang.VarDecl, mangled_name: ?[]co
     // does the same as:
     // extern int foo;
     // int foo = 2;
-    const is_extern = storage_class == .Extern and !has_init;
-    const is_export = !is_extern and storage_class != .Static;
+    var is_extern = storage_class == .Extern and !has_init;
+    var is_export = !is_extern and storage_class != .Static;
 
     const type_node = transQualTypeMaybeInitialized(c, qual_type, decl_init, var_decl_loc) catch |err| switch (err) {
         error.UnsupportedTranslation, error.UnsupportedType => {
@@ -656,7 +670,7 @@ fn visitVarDecl(c: *Context, var_decl: *const clang.VarDecl, mangled_name: ?[]co
     // If the initialization expression is not present, initialize with undefined.
     // If it is an integer literal, we can skip the @as since it will be redundant
     // with the variable type.
-    if (has_init) {
+    if (has_init) trans_init: {
         if (decl_init) |expr| {
             const node_or_error = if (expr.getStmtClass() == .StringLiteralClass)
                 transStringLiteralAsArray(c, scope, @ptrCast(*const clang.StringLiteral, expr), zigArraySize(c, type_node) catch 0)
@@ -666,7 +680,10 @@ fn visitVarDecl(c: *Context, var_decl: *const clang.VarDecl, mangled_name: ?[]co
                 error.UnsupportedTranslation,
                 error.UnsupportedType,
                 => {
-                    return failDecl(c, var_decl_loc, checked_name, "unable to translate initializer", .{});
+                    is_extern = true;
+                    is_export = false;
+                    try warn(c, scope, var_decl_loc, "unable to translate variable initializer, demoted to extern", .{});
+                    break :trans_init;
                 },
                 error.OutOfMemory => |e| return e,
             };
@@ -1845,7 +1862,7 @@ fn literalFitsInType(c: *Context, expr: *const clang.Expr, qt: clang.QualType) b
     var width = qualTypeIntBitWidth(c, qt) catch 8;
     if (width == 0) width = 8; // Byte is the smallest type.
     const is_signed = cIsSignedInteger(qt);
-    const width_max_int= (@as(u64, 1) << math.lossyCast(u6, width - @boolToInt(is_signed))) - 1;
+    const width_max_int = (@as(u64, 1) << math.lossyCast(u6, width - @boolToInt(is_signed))) - 1;
 
     switch (@ptrCast(*const clang.Stmt, expr).getStmtClass()) {
         .CharacterLiteralClass => {
@@ -1866,7 +1883,6 @@ fn literalFitsInType(c: *Context, expr: *const clang.Expr, qt: clang.QualType) b
         },
         else => unreachable,
     }
-
 }
 
 fn transInitListExprRecord(
test/translate_c.zig
@@ -11,12 +11,12 @@ pub fn addCases(cases: *tests.TranslateCContext) void {
         \\}
     });
 
-    cases.add("variadic function demoted to prototype",
+    cases.add("variadic function demoted to extern",
         \\int foo(int bar, ...) {
         \\    return 1;
         \\}
     , &[_][]const u8{
-        \\warning: TODO unable to translate variadic function, demoted to declaration
+        \\warning: TODO unable to translate variadic function, demoted to extern
         \\pub extern fn foo(bar: c_int, ...) c_int;
     });