mescc: Tinycc support: anonymous union.

* module/language/c99/compiler.mes (field:name): New function.
  (decl->info): Use it.
  (struct-field): Support anonymous union.
  (field:size): Update.
  (field-field): Update.
  (field-offset): Update.
This commit is contained in:
Jan Nieuwenhuizen 2017-07-22 20:40:41 +02:00
parent 2ab93d7308
commit 73726b2e27
3 changed files with 73 additions and 6 deletions

View File

@ -153,7 +153,8 @@ exec ${GUILE-guile} --no-auto-compile -L . -L guile -C . -C guile -s "$0" ${1+"$
"71-struct-array"
"72-typedef-struct-def"
"73-union"
"74-multi-line-string"))
"74-multi-line-string"
"75-struct-union"))
(add-target (group "check-scaffold-tests/7" #:dependencies (filter (target-prefix? "check-scaffold/tests/7") %targets)))

View File

@ -1105,8 +1105,17 @@
("unsigned long" . ,(make-type 'builtin 4 0 #f))
("unsigned long long" . ,(make-type 'builtin 8 0 #f))))
(define (field:name o)
(pmatch o
((union (,name ,type ,size ,pointer) . ,rest) name)
((union (,name ,type ,size) . ,rest) name)
((,name ,type ,size ,pointer) name)
((,name ,type ,size) name)
(_ (error "field:name not supported:" o))))
(define (field:size o)
(pmatch o
((union . ,fields) 4) ;; FIXME
((,name ,type ,size ,pointer) size)
((,name ,type ,size) size)
(_ 4)))
@ -1154,14 +1163,26 @@
(define (field-field info struct field)
(let* ((xtype (ast-type->type info struct))
(fields (type:description xtype)))
(and=> (member field fields (lambda (a b) (equal? a (car b)))) car)))
(let loop ((fields fields))
(if (null? fields) (error (format #f "no such field: ~a in ~s" field struct))
(let ((f (car fields)))
(cond ((equal? (car f) field) f)
((and (eq? (car f) 'union)
(find (lambda (x) (equal? (car x) field)) (cdr f))))
(else (loop (cdr fields)))))))))
(define (field-offset info struct field)
(let ((xtype (ast-type->type info struct)))
(if (eq? (type:type xtype) 'union) 0
(let* ((fields (type:description xtype))
(prefix (and=> (member field (reverse fields) (lambda (a b) (equal? a (car b)))) cdr)))
(apply + (map field:size prefix))))))
(let ((fields (type:description xtype)))
(let loop ((fields fields) (offset 0))
(if (null? fields) (error (format #f "no such field: ~a in ~s" field struct))
(let ((f (car fields)))
(cond ((equal? (car f) field) offset)
((and (eq? (car f) 'union)
(find (lambda (x) (equal? (car x) field)) (cdr f))
offset))
(else (loop (cdr fields) (+ offset (field:size f))))))))))))
(define (field-size info struct field)
(let ((xtype (ast-type->type info struct)))
@ -1384,6 +1405,9 @@
(let ((size (ast-type->size info `("struct" ,type))))
(list name type size 0)))
((comp-decl (decl-spec-list (type-spec (union-def (field-list . ,fields)))))
`(union ,@(map (struct-field info) fields)))
(_ (error "struct-field: unsupported: " o)))
)
)
@ -1772,7 +1796,7 @@
(empty (clone info #:text '())))
(let loop ((fields fields) (initzers initzers) (info info))
(if (null? fields) info
(let ((offset (field-offset info type (caar fields)))
(let ((offset (field-offset info type (field:name (car fields))))
(initzer (car initzers)))
(loop (cdr fields) (cdr initzers)
(clone info #:text

View File

@ -0,0 +1,42 @@
/* -*-comment-start: "//";comment-end:""-*-
* Mes --- Maxwell Equations of Software
* Copyright © 2017 Jan Nieuwenhuizen <janneke@gnu.org>
*
* This file is part of Mes.
*
* Mes is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or (at
* your option) any later version.
*
* Mes is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Mes. If not, see <http://www.gnu.org/licenses/>.
*/
#include "30-test.i"
#include <stdio.h>
struct foo
{
union {
int bar;
int baz;
};
};
int
test ()
{
struct foo f = {2};
printf ("f.bar=%d\n", f.bar);
if (f.bar != 2) return 1;
printf ("f.baz=%d\n", f.baz);
if (f.baz != 2) return 1;
return 0;
}