Commit 033cade6 authored by Rob Shearman's avatar Rob Shearman Committed by Alexandre Julliard

widl: Consolidate writing of COM and dispatch interfaces into one function to…

widl: Consolidate writing of COM and dispatch interfaces into one function to remove duplicated code. Split up the writing into start and end to eventually support the style MIDL uses where it writes declared types, etc. between the start and end of the interface. Make internal header functions take the file pointer to print to. Don't write interface IDs for non-object interfaces and always write handle declarations even if the interface has no methods, like MIDL does.
parent 550aaade
...@@ -620,11 +620,11 @@ const var_t *is_callas(const attr_list_t *a) ...@@ -620,11 +620,11 @@ const var_t *is_callas(const attr_list_t *a)
return get_attrp(a, ATTR_CALLAS); return get_attrp(a, ATTR_CALLAS);
} }
static void write_method_macro(const type_t *iface, const char *name) static void write_method_macro(FILE *header, const type_t *iface, const char *name)
{ {
const func_t *cur; const func_t *cur;
if (iface->ref) write_method_macro(iface->ref, name); if (iface->ref) write_method_macro(header, iface->ref, name);
if (!iface->funcs) return; if (!iface->funcs) return;
...@@ -683,7 +683,7 @@ void write_args(FILE *h, const var_list_t *args, const char *name, int method, i ...@@ -683,7 +683,7 @@ void write_args(FILE *h, const var_list_t *args, const char *name, int method, i
if (do_indent) indentation--; if (do_indent) indentation--;
} }
static void write_cpp_method_def(const type_t *iface) static void write_cpp_method_def(FILE *header, const type_t *iface)
{ {
const func_t *cur; const func_t *cur;
...@@ -708,11 +708,11 @@ static void write_cpp_method_def(const type_t *iface) ...@@ -708,11 +708,11 @@ static void write_cpp_method_def(const type_t *iface)
} }
} }
static void do_write_c_method_def(const type_t *iface, const char *name) static void do_write_c_method_def(FILE *header, const type_t *iface, const char *name)
{ {
const func_t *cur; const func_t *cur;
if (iface->ref) do_write_c_method_def(iface->ref, name); if (iface->ref) do_write_c_method_def(header, iface->ref, name);
if (!iface->funcs) return; if (!iface->funcs) return;
indent(header, 0); indent(header, 0);
...@@ -735,17 +735,17 @@ static void do_write_c_method_def(const type_t *iface, const char *name) ...@@ -735,17 +735,17 @@ static void do_write_c_method_def(const type_t *iface, const char *name)
} }
} }
static void write_c_method_def(const type_t *iface) static void write_c_method_def(FILE *header, const type_t *iface)
{ {
do_write_c_method_def(iface, iface->name); do_write_c_method_def(header, iface, iface->name);
} }
static void write_c_disp_method_def(const type_t *iface) static void write_c_disp_method_def(FILE *header, const type_t *iface)
{ {
do_write_c_method_def(iface->ref, iface->name); do_write_c_method_def(header, iface->ref, iface->name);
} }
static void write_method_proto(const type_t *iface) static void write_method_proto(FILE *header, const type_t *iface)
{ {
const func_t *cur; const func_t *cur;
...@@ -840,7 +840,7 @@ void write_locals(FILE *fp, const type_t *iface, int body) ...@@ -840,7 +840,7 @@ void write_locals(FILE *fp, const type_t *iface, int body)
} }
} }
static void write_function_proto(const type_t *iface, const func_t *fun, const char *prefix) static void write_function_proto(FILE *header, const type_t *iface, const func_t *fun, const char *prefix)
{ {
var_t *def = fun->def; var_t *def = fun->def;
const char *callconv = get_attrp(def->type->attrs, ATTR_CALLCONV); const char *callconv = get_attrp(def->type->attrs, ATTR_CALLCONV);
...@@ -858,7 +858,7 @@ static void write_function_proto(const type_t *iface, const func_t *fun, const c ...@@ -858,7 +858,7 @@ static void write_function_proto(const type_t *iface, const func_t *fun, const c
fprintf(header, ");\n\n"); fprintf(header, ");\n\n");
} }
static void write_function_protos(const type_t *iface) static void write_function_protos(FILE *header, const type_t *iface)
{ {
const func_t *cur; const func_t *cur;
int prefixes_differ = strcmp(prefix_client, prefix_server); int prefixes_differ = strcmp(prefix_client, prefix_server);
...@@ -868,10 +868,10 @@ static void write_function_protos(const type_t *iface) ...@@ -868,10 +868,10 @@ static void write_function_protos(const type_t *iface)
{ {
if (prefixes_differ) { if (prefixes_differ) {
fprintf(header, "/* client prototype */\n"); fprintf(header, "/* client prototype */\n");
write_function_proto(iface, cur, prefix_client); write_function_proto(header, iface, cur, prefix_client);
fprintf(header, "/* server prototype */\n"); fprintf(header, "/* server prototype */\n");
} }
write_function_proto(iface, cur, prefix_server); write_function_proto(header, iface, cur, prefix_server);
} }
} }
...@@ -892,48 +892,47 @@ void write_forward(type_t *iface) ...@@ -892,48 +892,47 @@ void write_forward(type_t *iface)
} }
} }
static void write_iface_guid(const type_t *iface) static void write_iface_guid(FILE *header, const type_t *iface)
{ {
const UUID *uuid = get_attrp(iface->attrs, ATTR_UUID); const UUID *uuid = get_attrp(iface->attrs, ATTR_UUID);
write_guid(header, "IID", iface->name, uuid); write_guid(header, "IID", iface->name, uuid);
} }
static void write_dispiface_guid(const type_t *iface) static void write_dispiface_guid(FILE *header, const type_t *iface)
{ {
const UUID *uuid = get_attrp(iface->attrs, ATTR_UUID); const UUID *uuid = get_attrp(iface->attrs, ATTR_UUID);
write_guid(header, "DIID", iface->name, uuid); write_guid(header, "DIID", iface->name, uuid);
} }
static void write_coclass_guid(type_t *cocl) static void write_coclass_guid(FILE *header, const type_t *cocl)
{ {
const UUID *uuid = get_attrp(cocl->attrs, ATTR_UUID); const UUID *uuid = get_attrp(cocl->attrs, ATTR_UUID);
write_guid(header, "CLSID", cocl->name, uuid); write_guid(header, "CLSID", cocl->name, uuid);
} }
static void write_com_interface(type_t *iface) static void write_com_interface_start(FILE *header, const type_t *iface)
{ {
if (!iface->funcs && !iface->ref) { int dispinterface = is_attr(iface->attrs, ATTR_DISPINTERFACE);
parser_warning("%s has no methods\n", iface->name);
return;
}
fprintf(header, "/*****************************************************************************\n"); fprintf(header, "/*****************************************************************************\n");
fprintf(header, " * %s interface\n", iface->name); fprintf(header, " * %s %sinterface\n", iface->name, dispinterface ? "disp" : "");
fprintf(header, " */\n"); fprintf(header, " */\n");
fprintf(header,"#ifndef __%s_INTERFACE_DEFINED__\n", iface->name); fprintf(header,"#ifndef __%s_%sINTERFACE_DEFINED__\n", iface->name, dispinterface ? "DISP" : "");
fprintf(header,"#define __%s_INTERFACE_DEFINED__\n\n", iface->name); fprintf(header,"#define __%s_%sINTERFACE_DEFINED__\n\n", iface->name, dispinterface ? "DISP" : "");
write_iface_guid(iface); }
write_forward(iface);
static void write_com_interface_end(FILE *header, type_t *iface)
{
int dispinterface = is_attr(iface->attrs, ATTR_DISPINTERFACE);
if (dispinterface)
write_dispiface_guid(header, iface);
else
write_iface_guid(header, iface);
/* C++ interface */ /* C++ interface */
fprintf(header, "#if defined(__cplusplus) && !defined(CINTERFACE)\n"); fprintf(header, "#if defined(__cplusplus) && !defined(CINTERFACE)\n");
if (iface->ref) if (iface->ref)
{ {
fprintf(header, "interface %s : public %s\n", iface->name, iface->ref->name); fprintf(header, "interface %s : public %s\n", iface->name, iface->ref->name);
fprintf(header, "{\n"); fprintf(header, "{\n");
indentation++;
write_cpp_method_def(iface);
indentation--;
fprintf(header, "};\n");
} }
else else
{ {
...@@ -941,19 +940,28 @@ static void write_com_interface(type_t *iface) ...@@ -941,19 +940,28 @@ static void write_com_interface(type_t *iface)
fprintf(header, "{\n"); fprintf(header, "{\n");
fprintf(header, " BEGIN_INTERFACE\n"); fprintf(header, " BEGIN_INTERFACE\n");
fprintf(header, "\n"); fprintf(header, "\n");
}
/* dispinterfaces don't have real functions, so don't write C++ functions for
* them */
if (!dispinterface)
{
indentation++; indentation++;
write_cpp_method_def(iface); write_cpp_method_def(header, iface);
indentation--; indentation--;
}
if (!iface->ref)
fprintf(header, " END_INTERFACE\n"); fprintf(header, " END_INTERFACE\n");
fprintf(header, "};\n"); fprintf(header, "};\n");
}
fprintf(header, "#else\n"); fprintf(header, "#else\n");
/* C interface */ /* C interface */
fprintf(header, "typedef struct %sVtbl {\n", iface->name); fprintf(header, "typedef struct %sVtbl {\n", iface->name);
indentation++; indentation++;
fprintf(header, " BEGIN_INTERFACE\n"); fprintf(header, " BEGIN_INTERFACE\n");
fprintf(header, "\n"); fprintf(header, "\n");
write_c_method_def(iface); if (dispinterface)
write_c_disp_method_def(header, iface);
else
write_c_method_def(header, iface);
indentation--; indentation--;
fprintf(header, " END_INTERFACE\n"); fprintf(header, " END_INTERFACE\n");
fprintf(header, "} %sVtbl;\n", iface->name); fprintf(header, "} %sVtbl;\n", iface->name);
...@@ -962,17 +970,25 @@ static void write_com_interface(type_t *iface) ...@@ -962,17 +970,25 @@ static void write_com_interface(type_t *iface)
fprintf(header, "};\n"); fprintf(header, "};\n");
fprintf(header, "\n"); fprintf(header, "\n");
fprintf(header, "#ifdef COBJMACROS\n"); fprintf(header, "#ifdef COBJMACROS\n");
write_method_macro(iface, iface->name); /* dispinterfaces don't have real functions, so don't write macros for them,
* only for the interface this interface inherits from, i.e. IDispatch */
write_method_macro(header, dispinterface ? iface->ref : iface, iface->name);
fprintf(header, "#endif\n"); fprintf(header, "#endif\n");
fprintf(header, "\n"); fprintf(header, "\n");
fprintf(header, "#endif\n"); fprintf(header, "#endif\n");
fprintf(header, "\n"); fprintf(header, "\n");
write_method_proto(iface); /* dispinterfaces don't have real functions, so don't write prototypes for
* them */
if (!dispinterface)
{
write_method_proto(header, iface);
write_locals(header, iface, FALSE); write_locals(header, iface, FALSE);
fprintf(header,"\n#endif /* __%s_INTERFACE_DEFINED__ */\n\n", iface->name); fprintf(header, "\n");
}
fprintf(header,"#endif /* __%s_%sINTERFACE_DEFINED__ */\n\n", iface->name, dispinterface ? "DISP" : "");
} }
static void write_rpc_interface(const type_t *iface) static void write_rpc_interface_start(FILE *header, const type_t *iface)
{ {
unsigned int ver = get_attrv(iface->attrs, ATTR_VERSION); unsigned int ver = get_attrv(iface->attrs, ATTR_VERSION);
const char *var = get_attrp(iface->attrs, ATTR_IMPLICIT_HANDLE); const char *var = get_attrp(iface->attrs, ATTR_IMPLICIT_HANDLE);
...@@ -990,9 +1006,6 @@ static void write_rpc_interface(const type_t *iface) ...@@ -990,9 +1006,6 @@ static void write_rpc_interface(const type_t *iface)
fprintf(header, " */\n"); fprintf(header, " */\n");
fprintf(header,"#ifndef __%s_INTERFACE_DEFINED__\n", iface->name); fprintf(header,"#ifndef __%s_INTERFACE_DEFINED__\n", iface->name);
fprintf(header,"#define __%s_INTERFACE_DEFINED__\n\n", iface->name); fprintf(header,"#define __%s_INTERFACE_DEFINED__\n\n", iface->name);
if (iface->funcs)
{
write_iface_guid(iface);
if (var) fprintf(header, "extern handle_t %s;\n", var); if (var) fprintf(header, "extern handle_t %s;\n", var);
if (old_names) if (old_names)
{ {
...@@ -1006,56 +1019,26 @@ static void write_rpc_interface(const type_t *iface) ...@@ -1006,56 +1019,26 @@ static void write_rpc_interface(const type_t *iface)
fprintf(header, "extern RPC_IF_HANDLE %s%s_v%d_%d_s_ifspec;\n", fprintf(header, "extern RPC_IF_HANDLE %s%s_v%d_%d_s_ifspec;\n",
prefix_server, iface->name, MAJORVERSION(ver), MINORVERSION(ver)); prefix_server, iface->name, MAJORVERSION(ver), MINORVERSION(ver));
} }
write_function_protos(iface);
}
fprintf(header,"\n#endif /* __%s_INTERFACE_DEFINED__ */\n\n", iface->name);
/* FIXME: server/client code */
} }
void write_interface(type_t *iface) static void write_rpc_interface_end(FILE *header, const type_t *iface)
{ {
if (is_object(iface->attrs)) fprintf(header,"\n#endif /* __%s_INTERFACE_DEFINED__ */\n\n", iface->name);
write_com_interface(iface);
else
write_rpc_interface(iface);
} }
void write_dispinterface(type_t *iface) void write_interface(type_t *iface)
{ {
fprintf(header, "/*****************************************************************************\n"); if (is_attr(iface->attrs, ATTR_DISPINTERFACE) || is_object(iface->attrs))
fprintf(header, " * %s dispinterface\n", iface->name); {
fprintf(header, " */\n"); write_com_interface_start(header, iface);
fprintf(header,"#ifndef __%s_DISPINTERFACE_DEFINED__\n", iface->name); write_com_interface_end(header, iface);
fprintf(header,"#define __%s_DISPINTERFACE_DEFINED__\n\n", iface->name); }
write_dispiface_guid(iface); else
write_forward(iface); {
/* C++ interface */ write_rpc_interface_start(header, iface);
fprintf(header, "#if defined(__cplusplus) && !defined(CINTERFACE)\n"); write_function_protos(header, iface);
fprintf(header, "interface %s : public %s\n", iface->name, iface->ref->name); write_rpc_interface_end(header, iface);
fprintf(header, "{\n"); }
fprintf(header, "};\n");
fprintf(header, "#else\n");
/* C interface */
fprintf(header, "typedef struct %sVtbl {\n", iface->name);
indentation++;
fprintf(header, " BEGIN_INTERFACE\n");
fprintf(header, "\n");
write_c_disp_method_def(iface);
indentation--;
fprintf(header, " END_INTERFACE\n");
fprintf(header, "} %sVtbl;\n", iface->name);
fprintf(header, "interface %s {\n", iface->name);
fprintf(header, " CONST_VTBL %sVtbl* lpVtbl;\n", iface->name);
fprintf(header, "};\n");
fprintf(header, "\n");
fprintf(header, "#ifdef COBJMACROS\n");
write_method_macro(iface->ref, iface->name);
fprintf(header, "#endif\n");
fprintf(header, "\n");
fprintf(header, "#endif\n");
fprintf(header, "\n");
fprintf(header,"#endif /* __%s_DISPINTERFACE_DEFINED__ */\n\n", iface->name);
} }
void write_coclass(type_t *cocl) void write_coclass(type_t *cocl)
...@@ -1063,7 +1046,7 @@ void write_coclass(type_t *cocl) ...@@ -1063,7 +1046,7 @@ void write_coclass(type_t *cocl)
fprintf(header, "/*****************************************************************************\n"); fprintf(header, "/*****************************************************************************\n");
fprintf(header, " * %s coclass\n", cocl->name); fprintf(header, " * %s coclass\n", cocl->name);
fprintf(header, " */\n\n"); fprintf(header, " */\n\n");
write_coclass_guid(cocl); write_coclass_guid(header, cocl);
fprintf(header, "\n"); fprintf(header, "\n");
} }
......
...@@ -52,7 +52,6 @@ extern void write_array(FILE *h, array_dims_t *v, int field); ...@@ -52,7 +52,6 @@ extern void write_array(FILE *h, array_dims_t *v, int field);
extern void write_import(const char *fname); extern void write_import(const char *fname);
extern void write_forward(type_t *iface); extern void write_forward(type_t *iface);
extern void write_interface(type_t *iface); extern void write_interface(type_t *iface);
extern void write_dispinterface(type_t *iface);
extern void write_locals(FILE *fp, const type_t *iface, int body); extern void write_locals(FILE *fp, const type_t *iface, int body);
extern void write_coclass(type_t *cocl); extern void write_coclass(type_t *cocl);
extern void write_coclass_forward(type_t *cocl); extern void write_coclass_forward(type_t *cocl);
......
...@@ -904,7 +904,7 @@ dispinterfacedef: dispinterfacehdr '{' ...@@ -904,7 +904,7 @@ dispinterfacedef: dispinterfacehdr '{'
'}' { $$ = $1; '}' { $$ = $1;
$$->fields_or_args = $3; $$->fields_or_args = $3;
$$->funcs = $4; $$->funcs = $4;
if (!parse_only && do_header) write_dispinterface($$); if (!parse_only && do_header) write_interface($$);
if (!parse_only && do_idfile) write_diid($$); if (!parse_only && do_idfile) write_diid($$);
is_in_interface = FALSE; is_in_interface = FALSE;
} }
...@@ -912,7 +912,7 @@ dispinterfacedef: dispinterfacehdr '{' ...@@ -912,7 +912,7 @@ dispinterfacedef: dispinterfacehdr '{'
'{' interface ';' '}' { $$ = $1; '{' interface ';' '}' { $$ = $1;
$$->fields_or_args = $3->fields_or_args; $$->fields_or_args = $3->fields_or_args;
$$->funcs = $3->funcs; $$->funcs = $3->funcs;
if (!parse_only && do_header) write_dispinterface($$); if (!parse_only && do_header) write_interface($$);
if (!parse_only && do_idfile) write_diid($$); if (!parse_only && do_idfile) write_diid($$);
is_in_interface = FALSE; is_in_interface = FALSE;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment