aboutsummaryrefslogtreecommitdiff
path: root/parser.y
diff options
context:
space:
mode:
authorFlavio Cruz <flaviocruz@gmail.com>2022-11-04 01:29:37 -0400
committerSamuel Thibault <samuel.thibault@ens-lyon.org>2022-11-05 22:21:04 +0100
commit3902cb2fcae6e2028252b5d2016bf0e99ed74980 (patch)
tree6f48ad0f9567569680529367b843e6e8abbece90 /parser.y
parent68b3d8fe3a9595b7a5cb2bb6bc5973ba26139704 (diff)
downloadmig-3902cb2fcae6e2028252b5d2016bf0e99ed74980.tar.gz
mig-3902cb2fcae6e2028252b5d2016bf0e99ed74980.tar.bz2
mig-3902cb2fcae6e2028252b5d2016bf0e99ed74980.zip
Add support to define structures in mig.
Basic syntax is presented below and allows users to define nested structured types by using simpler or structure types as members. Mig will use the C padding and alignment rules to produce the same size as the corresponding C structures. type timespec_t = struct { uint32_t tv_sec; uint32_t tv_nsec; }; This allows us to build stubs that are more easily adaptable to other architectures. Message-Id: <Y2SjQSMOINY8I5Dy@viriathus>
Diffstat (limited to 'parser.y')
-rw-r--r--parser.y46
1 files changed, 45 insertions, 1 deletions
diff --git a/parser.y b/parser.y
index 8521a84..104f604 100644
--- a/parser.y
+++ b/parser.y
@@ -84,6 +84,8 @@
%token syRAngle
%token syLBrack
%token syRBrack
+%token syLCBrack
+%token syRCBrack
%token syBar
%token syError /* lex error */
@@ -103,6 +105,7 @@
%type <statement_kind> ImportIndicant
%type <number> VarArrayHead ArrayHead StructHead IntExp
+%type <structured_type> StructList
%type <type> NamedTypeSpec TransTypeSpec TypeSpec
%type <type> CStringSpec
%type <type> BasicTypeSpec PrevTypeSpec ArgumentType
@@ -124,6 +127,7 @@
#include "type.h"
#include "routine.h"
#include "statement.h"
+#include "utils.h"
static const char *import_name(statement_kind_t sk);
@@ -149,6 +153,14 @@ yyerror(const char *s)
const_string_t outstr;
u_int size; /* 0 means there is no default size */
} symtype;
+ /* Holds information about a structure while parsing. */
+ struct
+ {
+ /* The required alignment (in bytes) so far. */
+ u_int type_alignment_in_bytes;
+ /* The size of the struct in bytes so far. */
+ u_int size_in_bytes;
+ } structured_type;
routine_t *routine;
arg_kind_t direction;
argument_t *argument;
@@ -466,11 +478,43 @@ TypeSpec : BasicTypeSpec
| syCaret TypeSpec
{ $$ = itPtrDecl($2); }
| StructHead TypeSpec
- { $$ = itStructDecl($1, $2); }
+ { $$ = itStructArrayDecl($1, $2); }
+ | syStruct syLCBrack StructList syRCBrack
+ { $$ = itStructDecl($3.size_in_bytes, $3.type_alignment_in_bytes); }
| CStringSpec
{ $$ = $1; }
;
+StructList : syIdentifier syIdentifier sySemi
+{
+ ipc_type_t *t = itPrevDecl($1);
+ if (!t) {
+ error("Type %s not found\n", $1);
+ }
+ if (!t->itInLine) {
+ error("Type %s must be inline\n", $2);
+ }
+
+ $$.type_alignment_in_bytes = t->itAlignment;
+ $$.size_in_bytes = t->itTypeSize;
+}
+ | StructList syIdentifier syIdentifier sySemi
+{
+ ipc_type_t *t = itPrevDecl($2);
+ if (!t) {
+ error("Type %s not found\n", $2);
+ }
+ if (!t->itInLine) {
+ error("Type %s must be inline\n", $2);
+ }
+ $$.type_alignment_in_bytes = MAX(t->itAlignment, $1.type_alignment_in_bytes);
+ int padding_bytes = 0;
+ if ($1.size_in_bytes % t->itAlignment)
+ padding_bytes = t->itAlignment - ($1.size_in_bytes % t->itAlignment);
+ $$.size_in_bytes = $1.size_in_bytes + padding_bytes + t->itTypeSize;
+}
+ ;
+
BasicTypeSpec : IPCType
{
$$ = itShortDecl($1.innumber, $1.instr,