Add enum support.

Another attempt at getting the alignment bits correct.
This commit is contained in:
Huw Davies 2005-01-25 20:02:09 +00:00 committed by Alexandre Julliard
parent a49278acc2
commit a43a250bd1
5 changed files with 62 additions and 49 deletions

View File

@ -428,15 +428,19 @@ enum_list: enum
enum: ident '=' expr_const { $$ = reg_const($1); enum: ident '=' expr_const { $$ = reg_const($1);
$$->eval = $3; $$->eval = $3;
$$->lval = $3->cval; $$->lval = $3->cval;
$$->type = make_type(RPC_FC_LONG, &std_int);
} }
| ident { $$ = reg_const($1); | ident { $$ = reg_const($1);
$$->lval = 0; /* default for first enum entry */ $$->lval = 0; /* default for first enum entry */
$$->type = make_type(RPC_FC_LONG, &std_int);
} }
; ;
enumdef: tENUM t_ident '{' enums '}' { $$ = get_typev(RPC_FC_ENUM16, $2, tsENUM); enumdef: tENUM t_ident '{' enums '}' { $$ = get_typev(RPC_FC_ENUM16, $2, tsENUM);
$$->fields = $4; $$->fields = $4;
$$->defined = TRUE; $$->defined = TRUE;
if(in_typelib)
add_enum($$);
} }
; ;

View File

@ -250,3 +250,16 @@ void add_struct(type_t *structure)
LINK(entry, typelib->entry); LINK(entry, typelib->entry);
typelib->entry = entry; typelib->entry = entry;
} }
void add_enum(type_t *enumeration)
{
typelib_entry_t *entry;
if (!typelib) return;
chat("add enum: %s\n", enumeration->name);
entry = xmalloc(sizeof(*entry));
entry->kind = TKIND_ENUM;
entry->u.enumeration = enumeration;
LINK(entry, typelib->entry);
typelib->entry = entry;
}

View File

@ -28,6 +28,7 @@ extern void add_interface(type_t *iface);
extern void add_coclass(class_t *cls); extern void add_coclass(class_t *cls);
extern void add_module(type_t *module); extern void add_module(type_t *module);
extern void add_struct(type_t *structure); extern void add_struct(type_t *structure);
extern void add_enum(type_t *enumeration);
/* Copied from wtypes.h. Not included directly because that would create a /* Copied from wtypes.h. Not included directly because that would create a
* circular dependency (after all, wtypes.h is generated by widl...) */ * circular dependency (after all, wtypes.h is generated by widl...) */

View File

@ -240,6 +240,7 @@ struct _typelib_entry_t {
type_t *interface; type_t *interface;
type_t *module; type_t *module;
type_t *structure; type_t *structure;
type_t *enumeration;
} u; } u;
DECL_LINK(typelib_entry_t) DECL_LINK(typelib_entry_t)
}; };

View File

@ -1353,9 +1353,9 @@ static HRESULT add_func_desc(msft_typeinfo_t* typeinfo, func_t *func, int index)
typeinfo->names[index] = offset; typeinfo->names[index] = offset;
namedata = typeinfo->typelib->typelib_segment_data[MSFT_SEG_NAME] + offset; namedata = typeinfo->typelib->typelib_segment_data[MSFT_SEG_NAME] + offset;
namedata[9] &= ~0x10;
if (*((INT *)namedata) == -1) { if (*((INT *)namedata) == -1) {
*((INT *)namedata) = typeinfo->typelib->typelib_typeinfo_offsets[typeinfo->typeinfo->typekind >> 16]; *((INT *)namedata) = typeinfo->typelib->typelib_typeinfo_offsets[typeinfo->typeinfo->typekind >> 16];
namedata[9] &= ~0x10;
} }
for (arg = last_arg, i = 0; arg; arg = PREV_LINK(arg), i++) { for (arg = last_arg, i = 0; arg; arg = PREV_LINK(arg), i++) {
@ -1369,49 +1369,13 @@ static HRESULT add_func_desc(msft_typeinfo_t* typeinfo, func_t *func, int index)
return S_OK; return S_OK;
} }
static void set_alignment(
msft_typeinfo_t* typeinfo,
WORD cbAlignment)
{
if (!cbAlignment) return;
if (cbAlignment > 16) return;
typeinfo->typeinfo->typekind &= ~0xf800;
/* FIXME: There's probably some way to simplify this. */
switch (typeinfo->typeinfo->typekind & 15) {
case TKIND_ALIAS:
default:
break;
case TKIND_ENUM:
case TKIND_INTERFACE:
case TKIND_DISPATCH:
case TKIND_COCLASS:
if (cbAlignment > 4) cbAlignment = 4;
break;
case TKIND_RECORD:
case TKIND_MODULE:
case TKIND_UNION:
cbAlignment = 1;
break;
}
typeinfo->typeinfo->typekind |= cbAlignment << 11;
return;
}
static HRESULT add_var_desc(msft_typeinfo_t *typeinfo, UINT index, var_t* var) static HRESULT add_var_desc(msft_typeinfo_t *typeinfo, UINT index, var_t* var)
{ {
int offset; int offset;
INT *typedata; INT *typedata;
int var_datawidth; int var_datawidth;
int var_alignment; int var_alignment;
int var_type_size; int var_type_size, var_kind = 0 /* VAR_PERINSTANCE */;
int alignment; int alignment;
int varflags = 0; int varflags = 0;
attr_t *attr; attr_t *attr;
@ -1460,20 +1424,26 @@ static HRESULT add_var_desc(msft_typeinfo_t *typeinfo, UINT index, var_t* var)
/* pad out starting position to data width */ /* pad out starting position to data width */
typeinfo->datawidth += var_alignment - 1; typeinfo->datawidth += var_alignment - 1;
typeinfo->datawidth &= ~(var_alignment - 1); typeinfo->datawidth &= ~(var_alignment - 1);
typedata[4] = typeinfo->datawidth;
/* add the new variable to the total data width */ if((typeinfo->typeinfo->typekind & 0xf) == TKIND_ENUM) {
write_value(typeinfo->typelib, &typedata[4], VT_I4, &var->lval);
var_kind = 2; /* VAR_CONST */
var_type_size += 16; /* sizeof(VARIANT) */
typeinfo->datawidth = var_datawidth;
} else {
typedata[4] = typeinfo->datawidth;
typeinfo->datawidth += var_datawidth; typeinfo->datawidth += var_datawidth;
}
/* add type description size to total required allocation */ /* add type description size to total required allocation */
typedata[3] += var_type_size << 16; typedata[3] += var_type_size << 16 | var_kind;
/* fix type alignment */ /* fix type alignment */
alignment = (typeinfo->typeinfo->typekind >> 11) & 0x1f; alignment = (typeinfo->typeinfo->typekind >> 11) & 0x1f;
if (alignment < var_alignment) { if (alignment < var_alignment) {
alignment = var_alignment; alignment = var_alignment;
typeinfo->typeinfo->typekind &= ~0xf800; typeinfo->typeinfo->typekind &= ~0xffc0;
typeinfo->typeinfo->typekind |= alignment << 11; typeinfo->typeinfo->typekind |= alignment << 11 | alignment << 6;
} }
/* ??? */ /* ??? */
@ -1499,7 +1469,9 @@ static HRESULT add_var_desc(msft_typeinfo_t *typeinfo, UINT index, var_t* var)
if (*((INT *)namedata) == -1) { if (*((INT *)namedata) == -1) {
*((INT *)namedata) = typeinfo->typelib->typelib_typeinfo_offsets[typeinfo->typeinfo->typekind >> 16]; *((INT *)namedata) = typeinfo->typelib->typelib_typeinfo_offsets[typeinfo->typeinfo->typekind >> 16];
namedata[9] |= 0x10; namedata[9] |= 0x10;
} } else
namedata[9] &= ~0x10;
if ((typeinfo->typeinfo->typekind & 15) == TKIND_ENUM) { if ((typeinfo->typeinfo->typekind & 15) == TKIND_ENUM) {
namedata[9] |= 0x20; namedata[9] |= 0x20;
} }
@ -1544,8 +1516,7 @@ static msft_typeinfo_t *create_msft_typeinfo(msft_typelib_t *typelib, enum type_
msft_typeinfo->typeinfo = typeinfo; msft_typeinfo->typeinfo = typeinfo;
typeinfo->typekind |= kind | 0x220; typeinfo->typekind |= kind | 0x20;
set_alignment(msft_typeinfo, 4);
for( ; attr; attr = NEXT_LINK(attr)) { for( ; attr; attr = NEXT_LINK(attr)) {
switch(attr->type) { switch(attr->type) {
@ -1620,6 +1591,7 @@ static void add_interface_typeinfo(msft_typelib_t *typelib, type_t *interface)
msft_typeinfo = create_msft_typeinfo(typelib, TKIND_INTERFACE, interface->name, interface->attrs, msft_typeinfo = create_msft_typeinfo(typelib, TKIND_INTERFACE, interface->name, interface->attrs,
typelib->typelib_header.nrtypeinfos); typelib->typelib_header.nrtypeinfos);
msft_typeinfo->typeinfo->size = 4; msft_typeinfo->typeinfo->size = 4;
msft_typeinfo->typeinfo->typekind |= 0x2200;
if(interface->ref) if(interface->ref)
add_impl_type(msft_typeinfo, interface->ref); add_impl_type(msft_typeinfo, interface->ref);
@ -1651,6 +1623,25 @@ static void add_structure_typeinfo(msft_typelib_t *typelib, type_t *structure)
} }
} }
static void add_enum_typeinfo(msft_typelib_t *typelib, type_t *enumeration)
{
int idx = 0;
var_t *cur = enumeration->fields;
msft_typeinfo_t *msft_typeinfo;
enumeration->typelib_idx = typelib->typelib_header.nrtypeinfos;
msft_typeinfo = create_msft_typeinfo(typelib, TKIND_ENUM, enumeration->name, enumeration->attrs,
typelib->typelib_header.nrtypeinfos);
msft_typeinfo->typeinfo->size = 0;
while(NEXT_LINK(cur)) cur = NEXT_LINK(cur);
while(cur) {
add_var_desc(msft_typeinfo, idx, cur);
idx++;
cur = PREV_LINK(cur);
}
}
static void add_entry(msft_typelib_t *typelib, typelib_entry_t *entry) static void add_entry(msft_typelib_t *typelib, typelib_entry_t *entry)
{ {
switch(entry->kind) { switch(entry->kind) {
@ -1662,13 +1653,16 @@ static void add_entry(msft_typelib_t *typelib, typelib_entry_t *entry)
add_structure_typeinfo(typelib, entry->u.structure); add_structure_typeinfo(typelib, entry->u.structure);
break; break;
case TKIND_ENUM:
add_enum_typeinfo(typelib, entry->u.enumeration);
break;
default: default:
error("add_entry: unhandled type %d\n", entry->kind); error("add_entry: unhandled type %d\n", entry->kind);
break; break;
} }
} }
static void set_name(msft_typelib_t *typelib) static void set_name(msft_typelib_t *typelib)
{ {
int offset; int offset;