diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2020-04-08 14:11:16 -0400 |
|---|---|---|
| committer | Andrew Kelley <andrew@ziglang.org> | 2020-04-08 14:11:16 -0400 |
| commit | 29ea425ca20305a427919b9a1df85a5c5a8657c3 (patch) | |
| tree | 244557b0172b781f1dcaa6b2538cf9e515943576 | |
| parent | b3aef49eeaa42127ac57ded3b15228db39f806b3 (diff) | |
| parent | d7902707bcc1faede9ef5490d02bcea935e3b8fc (diff) | |
| download | zig-29ea425ca20305a427919b9a1df85a5c5a8657c3.tar.gz zig-29ea425ca20305a427919b9a1df85a5c5a8657c3.zip | |
Merge branch 'LakeByTheWoods-translate_ptr_int'
closes #4688
| -rw-r--r-- | src-self-hosted/translate_c.zig | 81 | ||||
| -rw-r--r-- | test/translate_c.zig | 39 |
2 files changed, 96 insertions, 24 deletions
diff --git a/src-self-hosted/translate_c.zig b/src-self-hosted/translate_c.zig index e115580244..e4d8cc4397 100644 --- a/src-self-hosted/translate_c.zig +++ b/src-self-hosted/translate_c.zig @@ -5479,18 +5479,20 @@ fn parseCPrimaryExpr(c: *Context, it: *CTokenList.Iterator, source: []const u8, .LParen => { const inner_node = try parseCExpr(c, it, source, source_loc, scope); - if (it.next().?.id != .RParen) { + const next_id = it.next().?.id; + if (next_id != .RParen) { const first_tok = it.list.at(0); try failDecl( c, source_loc, source[first_tok.start..first_tok.end], - "unable to translate C expr: expected ')'' here", - .{}, + "unable to translate C expr: expected ')'' instead got: {}", + .{@tagName(next_id)}, ); return error.ParseError; } var saw_l_paren = false; + var saw_integer_literal = false; switch (it.peek().?.id) { // (type)(to_cast) .LParen => { @@ -5499,6 +5501,10 @@ fn parseCPrimaryExpr(c: *Context, it: *CTokenList.Iterator, source: []const u8, }, // (type)identifier .Identifier => {}, + // (type)integer + .IntegerLiteral => { + saw_integer_literal = true; + }, else => return inner_node, } @@ -5519,12 +5525,21 @@ fn parseCPrimaryExpr(c: *Context, it: *CTokenList.Iterator, source: []const u8, return error.ParseError; } - //if (@typeInfo(@TypeOf(x)) == .Pointer) - // @ptrCast(dest, x) - //else if (@typeInfo(@TypeOf(x)) == .Int and @typeInfo(dest) == .Pointer) + if (saw_integer_literal) { + // @intToPtr(dest, x) + const int_to_ptr = try transCreateNodeBuiltinFnCall(c, "@intToPtr"); + try int_to_ptr.params.push(inner_node); + try int_to_ptr.params.push(node_to_cast); + int_to_ptr.rparen_token = try appendToken(c, .RParen, ")"); + return &int_to_ptr.base; + } + + //( if (@typeInfo(@TypeOf(x)) == .Pointer) + // @ptrCast(dest, @alignCast(@alignOf(dest.Child), x)) + //else if (@typeInfo(@TypeOf(x)) == .Integer and @typeInfo(dest) == .Pointer)) // @intToPtr(dest, x) //else - // @as(dest, x) + // @as(dest, x) ) const lparen = try appendToken(c, .LParen, "("); @@ -5546,9 +5561,30 @@ fn parseCPrimaryExpr(c: *Context, it: *CTokenList.Iterator, source: []const u8, if_1.condition = &cmp_1.base; _ = try appendToken(c, .RParen, ")"); + const period_tok = try appendToken(c, .Period, "."); + const child_ident = try transCreateNodeIdentifier(c, "Child"); + const inner_node_child = try c.a().create(ast.Node.InfixOp); + inner_node_child.* = .{ + .op_token = period_tok, + .lhs = inner_node, + .op = .Period, + .rhs = child_ident, + }; + + const align_of = try transCreateNodeBuiltinFnCall(c, "@alignOf"); + try align_of.params.push(&inner_node_child.base); + align_of.rparen_token = try appendToken(c, .RParen, ")"); + // hack to get zig fmt to render a comma in builtin calls + _ = try appendToken(c, .Comma, ","); + + const align_cast = try transCreateNodeBuiltinFnCall(c, "@alignCast"); + try align_cast.params.push(&align_of.base); + try align_cast.params.push(node_to_cast); + align_cast.rparen_token = try appendToken(c, .RParen, ")"); + const ptr_cast = try transCreateNodeBuiltinFnCall(c, "@ptrCast"); try ptr_cast.params.push(inner_node); - try ptr_cast.params.push(node_to_cast); + try ptr_cast.params.push(&align_cast.base); ptr_cast.rparen_token = try appendToken(c, .RParen, ")"); if_1.body = &ptr_cast.base; @@ -5729,19 +5765,24 @@ fn parseCSuffixOpExpr(c: *Context, it: *CTokenList.Iterator, source: []const u8, // hack to get zig fmt to render a comma in builtin calls _ = try appendToken(c, .Comma, ","); - const ptr_kind = blk: { - // * token - _ = it.prev(); - // last token of `node` - const prev_id = it.prev().?.id; - _ = it.next(); - _ = it.next(); - break :blk if (prev_id == .Keyword_void) .Asterisk else Token.Id.Identifier; - }; + // * token + _ = it.prev(); + // last token of `node` + const prev_id = it.prev().?.id; + _ = it.next(); + _ = it.next(); - const ptr = try transCreateNodePtrType(c, false, false, ptr_kind); - ptr.rhs = node; - return &ptr.base; + if (prev_id == .Keyword_void) { + const ptr = try transCreateNodePtrType(c, false, false, .Asterisk); + ptr.rhs = node; + const optional_node = try transCreateNodePrefixOp(c, .OptionalType, .QuestionMark, "?"); + optional_node.rhs = &ptr.base; + return &optional_node.base; + } else { + const ptr = try transCreateNodePtrType(c, false, false, Token.Id.Identifier); + ptr.rhs = node; + return &ptr.base; + } } else { // expr * expr op_token = try appendToken(c, .Asterisk, "*"); diff --git a/test/translate_c.zig b/test/translate_c.zig index 55414488f6..9e935638e6 100644 --- a/test/translate_c.zig +++ b/test/translate_c.zig @@ -1458,7 +1458,7 @@ pub fn addCases(cases: *tests.TranslateCContext) void { cases.add("macro pointer cast", \\#define NRF_GPIO ((NRF_GPIO_Type *) NRF_GPIO_BASE) , &[_][]const u8{ - \\pub const NRF_GPIO = (if (@typeInfo(@TypeOf(NRF_GPIO_BASE)) == .Pointer) @ptrCast([*c]NRF_GPIO_Type, NRF_GPIO_BASE) else if (@typeInfo(@TypeOf(NRF_GPIO_BASE)) == .Int and @typeInfo([*c]NRF_GPIO_Type) == .Pointer) @intToPtr([*c]NRF_GPIO_Type, NRF_GPIO_BASE) else @as([*c]NRF_GPIO_Type, NRF_GPIO_BASE)); + \\pub const NRF_GPIO = (if (@typeInfo(@TypeOf(NRF_GPIO_BASE)) == .Pointer) @ptrCast([*c]NRF_GPIO_Type, @alignCast(@alignOf([*c]NRF_GPIO_Type.Child), NRF_GPIO_BASE)) else if (@typeInfo(@TypeOf(NRF_GPIO_BASE)) == .Int and @typeInfo([*c]NRF_GPIO_Type) == .Pointer) @intToPtr([*c]NRF_GPIO_Type, NRF_GPIO_BASE) else @as([*c]NRF_GPIO_Type, NRF_GPIO_BASE)); }); cases.add("basic macro function", @@ -2668,11 +2668,11 @@ pub fn addCases(cases: *tests.TranslateCContext) void { \\#define FOO(bar) baz((void *)(baz)) \\#define BAR (void*) a , &[_][]const u8{ - \\pub inline fn FOO(bar: var) @TypeOf(baz((if (@typeInfo(@TypeOf(baz)) == .Pointer) @ptrCast(*c_void, baz) else if (@typeInfo(@TypeOf(baz)) == .Int and @typeInfo(*c_void) == .Pointer) @intToPtr(*c_void, baz) else @as(*c_void, baz)))) { - \\ return baz((if (@typeInfo(@TypeOf(baz)) == .Pointer) @ptrCast(*c_void, baz) else if (@typeInfo(@TypeOf(baz)) == .Int and @typeInfo(*c_void) == .Pointer) @intToPtr(*c_void, baz) else @as(*c_void, baz))); + \\pub inline fn FOO(bar: var) @TypeOf(baz((if (@typeInfo(@TypeOf(baz)) == .Pointer) @ptrCast(?*c_void, @alignCast(@alignOf(?*c_void.Child), baz)) else if (@typeInfo(@TypeOf(baz)) == .Int and @typeInfo(?*c_void) == .Pointer) @intToPtr(?*c_void, baz) else @as(?*c_void, baz)))) { + \\ return baz((if (@typeInfo(@TypeOf(baz)) == .Pointer) @ptrCast(?*c_void, @alignCast(@alignOf(?*c_void.Child), baz)) else if (@typeInfo(@TypeOf(baz)) == .Int and @typeInfo(?*c_void) == .Pointer) @intToPtr(?*c_void, baz) else @as(?*c_void, baz))); \\} , - \\pub const BAR = (if (@typeInfo(@TypeOf(a)) == .Pointer) @ptrCast(*c_void, a) else if (@typeInfo(@TypeOf(a)) == .Int and @typeInfo(*c_void) == .Pointer) @intToPtr(*c_void, a) else @as(*c_void, a)); + \\pub const BAR = (if (@typeInfo(@TypeOf(a)) == .Pointer) @ptrCast(?*c_void, @alignCast(@alignOf(?*c_void.Child), a)) else if (@typeInfo(@TypeOf(a)) == .Int and @typeInfo(?*c_void) == .Pointer) @intToPtr(?*c_void, a) else @as(?*c_void, a)); }); cases.add("macro conditional operator", @@ -2879,4 +2879,35 @@ pub fn addCases(cases: *tests.TranslateCContext) void { , &[_][]const u8{ \\pub const FOO = 0x61626364; }); + + cases.add("Make sure casts are grouped", + \\typedef struct + \\{ + \\ int i; + \\} + \\*_XPrivDisplay; + \\typedef struct _XDisplay Display; + \\#define DefaultScreen(dpy) (((_XPrivDisplay)(dpy))->default_screen) + \\ + , &[_][]const u8{ + \\pub inline fn DefaultScreen(dpy: var) @TypeOf((if (@typeInfo(@TypeOf(dpy)) == .Pointer) @ptrCast(_XPrivDisplay, @alignCast(@alignOf(_XPrivDisplay.Child), dpy)) else if (@typeInfo(@TypeOf(dpy)) == .Int and @typeInfo(_XPrivDisplay) == .Pointer) @intToPtr(_XPrivDisplay, dpy) else @as(_XPrivDisplay, dpy)).*.default_screen) { + \\ return (if (@typeInfo(@TypeOf(dpy)) == .Pointer) @ptrCast(_XPrivDisplay, @alignCast(@alignOf(_XPrivDisplay.Child), dpy)) else if (@typeInfo(@TypeOf(dpy)) == .Int and @typeInfo(_XPrivDisplay) == .Pointer) @intToPtr(_XPrivDisplay, dpy) else @as(_XPrivDisplay, dpy)).*.default_screen; + \\} + }); + + cases.add("Cast from integer literals to poiter", + \\#define NULL ((void*)0) + \\#define GPIO_0_MEM_MAP ((unsigned*)0x8000) + \\#define GPIO_1_MEM_MAP ((unsigned*)0x8004) + \\#define GPIO_2_MEM_MAP ((unsigned*)0x8008) + \\ + , &[_][]const u8{ + \\pub const NULL = @intToPtr(?*c_void, 0); + , + \\pub const GPIO_0_MEM_MAP = @intToPtr([*c]c_uint, 0x8000); + , + \\pub const GPIO_1_MEM_MAP = @intToPtr([*c]c_uint, 0x8004); + , + \\pub const GPIO_2_MEM_MAP = @intToPtr([*c]c_uint, 0x8008); + }); } |
