xref: /reactos/dll/win32/rpcrt4/rpc_epmap.c (revision ebaf247c)
1 /*
2  * RPC endpoint mapper
3  *
4  * Copyright 2002 Greg Turner
5  * Copyright 2001 Ove Kåven, TransGaming Technologies
6  * Copyright 2008 Robert Shearman (for CodeWeavers)
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Lesser General Public
10  * License as published by the Free Software Foundation; either
11  * version 2.1 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public
19  * License along with this library; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21  */
22 
23 #include <stdarg.h>
24 
25 #include "windef.h"
26 #include "winbase.h"
27 #include "winerror.h"
28 #include "winsvc.h"
29 
30 #include "rpc.h"
31 
32 #include "wine/debug.h"
33 #include "wine/exception.h"
34 
35 #include "rpc_binding.h"
36 #include "epm_c.h"
37 #include "epm_towers.h"
38 
39 WINE_DEFAULT_DEBUG_CHANNEL(ole);
40 
41 /* The "real" RPC portmapper endpoints that I know of are:
42  *
43  *  ncadg_ip_udp: 135
44  *  ncacn_ip_tcp: 135
45  *  ncacn_np: \\pipe\epmapper
46  *  ncalrpc: epmapper
47  *  ncacn_http: 593
48  *
49  * If the user's machine ran a DCE RPC daemon, it would
50  * probably be possible to connect to it, but there are many
51  * reasons not to, like:
52  *  - the user probably does *not* run one, and probably
53  *    shouldn't be forced to run one just for local COM
54  *  - very few Unix systems use DCE RPC... if they run a RPC
55  *    daemon at all, it's usually Sun RPC
56  *  - DCE RPC registrations are persistent and saved on disk,
57  *    while MS-RPC registrations are documented as non-persistent
58  *    and stored only in RAM, and auto-destroyed when the process
59  *    dies (something DCE RPC can't do)
60  *
61  * Of course, if the user *did* want to run a DCE RPC daemon anyway,
62  * there would be interoperability advantages, like the possibility
63  * of running a fully functional DCOM server using Wine...
64  */
65 
66 static const struct epm_endpoints
67 {
68     const char *protseq;
69     const char *endpoint;
70 } epm_endpoints[] =
71 {
72     { "ncacn_np", "\\pipe\\epmapper" },
73     { "ncacn_ip_tcp", "135" },
74     { "ncacn_ip_udp", "135" },
75     { "ncalrpc", "epmapper" },
76     { "ncacn_http", "593" },
77 };
78 
79 static BOOL start_rpcss(void)
80 {
81     static const WCHAR rpcssW[] = {'R','p','c','S','s',0};
82     SC_HANDLE scm, service;
83     SERVICE_STATUS_PROCESS status;
84     BOOL ret = FALSE;
85 
86     TRACE("\n");
87 
88     if (!(scm = OpenSCManagerW( NULL, NULL, 0 )))
89     {
90         ERR( "failed to open service manager\n" );
91         return FALSE;
92     }
93     if (!(service = OpenServiceW( scm, rpcssW, SERVICE_START | SERVICE_QUERY_STATUS )))
94     {
95         ERR( "failed to open RpcSs service\n" );
96         CloseServiceHandle( scm );
97         return FALSE;
98     }
99     if (StartServiceW( service, 0, NULL ) || GetLastError() == ERROR_SERVICE_ALREADY_RUNNING)
100     {
101         ULONGLONG start_time = GetTickCount64();
102         do
103         {
104             DWORD dummy;
105 
106             if (!QueryServiceStatusEx( service, SC_STATUS_PROCESS_INFO,
107                                        (BYTE *)&status, sizeof(status), &dummy ))
108                 break;
109             if (status.dwCurrentState == SERVICE_RUNNING)
110             {
111                 ret = TRUE;
112                 break;
113             }
114             if (GetTickCount64() - start_time > 30000) break;
115             Sleep( 100 );
116 
117         } while (status.dwCurrentState == SERVICE_START_PENDING);
118 
119         if (status.dwCurrentState != SERVICE_RUNNING)
120             WARN( "RpcSs failed to start %u\n", status.dwCurrentState );
121     }
122     else ERR( "failed to start RpcSs service\n" );
123 
124     CloseServiceHandle( service );
125     CloseServiceHandle( scm );
126     return ret;
127 }
128 
129 static inline BOOL is_epm_destination_local(RPC_BINDING_HANDLE handle)
130 {
131     RpcBinding *bind = handle;
132     const char *protseq = bind->Protseq;
133     const char *network_addr = bind->NetworkAddr;
134 
135     return (!strcmp(protseq, "ncalrpc") ||
136            (!strcmp(protseq, "ncacn_np") &&
137                 (!network_addr || !strcmp(network_addr, "."))));
138 }
139 
140 static RPC_STATUS get_epm_handle_client(RPC_BINDING_HANDLE handle, RPC_BINDING_HANDLE *epm_handle)
141 {
142     RpcBinding *bind = handle;
143     const char * pszEndpoint = NULL;
144     RPC_STATUS status;
145     RpcBinding* epm_bind;
146     unsigned int i;
147 
148     if (bind->server)
149         return RPC_S_INVALID_BINDING;
150 
151     for (i = 0; i < ARRAY_SIZE(epm_endpoints); i++)
152         if (!strcmp(bind->Protseq, epm_endpoints[i].protseq))
153             pszEndpoint = epm_endpoints[i].endpoint;
154 
155     if (!pszEndpoint)
156     {
157         FIXME("no endpoint for the endpoint-mapper found for protseq %s\n", debugstr_a(bind->Protseq));
158         return RPC_S_PROTSEQ_NOT_SUPPORTED;
159     }
160 
161     status = RpcBindingCopy(handle, epm_handle);
162     if (status != RPC_S_OK) return status;
163 
164     epm_bind = *epm_handle;
165     if (epm_bind->AuthInfo)
166     {
167         /* don't bother with authenticating against the EPM by default
168         * (see EnableAuthEpResolution registry value) */
169         RpcAuthInfo_Release(epm_bind->AuthInfo);
170         epm_bind->AuthInfo = NULL;
171     }
172     RPCRT4_ResolveBinding(epm_bind, pszEndpoint);
173     TRACE("RPC_S_OK\n");
174     return RPC_S_OK;
175 }
176 
177 static RPC_STATUS get_epm_handle_server(RPC_BINDING_HANDLE *epm_handle)
178 {
179     unsigned char string_binding[] = "ncacn_np:.[\\\\pipe\\\\epmapper]";
180 
181     return RpcBindingFromStringBindingA(string_binding, epm_handle);
182 }
183 
184 static LONG WINAPI rpc_filter(EXCEPTION_POINTERS *__eptr)
185 {
186     switch (__eptr->ExceptionRecord->ExceptionCode)
187     {
188         case EXCEPTION_ACCESS_VIOLATION:
189         case EXCEPTION_ILLEGAL_INSTRUCTION:
190             return EXCEPTION_CONTINUE_SEARCH;
191         default:
192             return EXCEPTION_EXECUTE_HANDLER;
193     }
194 }
195 
196 static RPC_STATUS epm_register( RPC_IF_HANDLE IfSpec, RPC_BINDING_VECTOR *BindingVector,
197                                 UUID_VECTOR *UuidVector, RPC_CSTR Annotation, BOOL replace )
198 {
199   PRPC_SERVER_INTERFACE If = IfSpec;
200   ULONG i;
201   RPC_STATUS status = RPC_S_OK;
202   error_status_t status2;
203   ept_entry_t *entries;
204   handle_t handle;
205 
206   TRACE("(%p,%p,%p,%s) replace=%d\n", IfSpec, BindingVector, UuidVector, debugstr_a((char*)Annotation), replace);
207   TRACE(" ifid=%s\n", debugstr_guid(&If->InterfaceId.SyntaxGUID));
208   for (i=0; i<BindingVector->Count; i++) {
209     RpcBinding* bind = BindingVector->BindingH[i];
210     TRACE(" protseq[%d]=%s\n", i, debugstr_a(bind->Protseq));
211     TRACE(" endpoint[%d]=%s\n", i, debugstr_a(bind->Endpoint));
212   }
213   if (UuidVector) {
214     for (i=0; i<UuidVector->Count; i++)
215       TRACE(" obj[%d]=%s\n", i, debugstr_guid(UuidVector->Uuid[i]));
216   }
217 
218   if (!BindingVector->Count) return RPC_S_OK;
219 
220   entries = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*entries) * BindingVector->Count * (UuidVector ? UuidVector->Count : 1));
221   if (!entries)
222       return RPC_S_OUT_OF_MEMORY;
223 
224   status = get_epm_handle_server(&handle);
225   if (status != RPC_S_OK)
226   {
227     HeapFree(GetProcessHeap(), 0, entries);
228     return status;
229   }
230 
231   for (i = 0; i < BindingVector->Count; i++)
232   {
233       unsigned j;
234       RpcBinding* bind = BindingVector->BindingH[i];
235       for (j = 0; j < (UuidVector ? UuidVector->Count : 1); j++)
236       {
237           status = TowerConstruct(&If->InterfaceId, &If->TransferSyntax,
238                                   bind->Protseq, bind->Endpoint,
239                                   bind->NetworkAddr,
240                                   &entries[i*(UuidVector ? UuidVector->Count : 1) + j].tower);
241           if (status != RPC_S_OK) break;
242 
243           if (UuidVector)
244               memcpy(&entries[i * UuidVector->Count].object, &UuidVector->Uuid[j], sizeof(GUID));
245           else
246               memset(&entries[i].object, 0, sizeof(entries[i].object));
247           if (Annotation)
248               memcpy(entries[i].annotation, Annotation,
249                      min(strlen((char *)Annotation) + 1, ept_max_annotation_size));
250       }
251   }
252 
253   if (status == RPC_S_OK)
254   {
255       while (TRUE)
256       {
257           __TRY
258           {
259               ept_insert(handle, BindingVector->Count * (UuidVector ? UuidVector->Count : 1),
260                          entries, replace, &status2);
261           }
262           __EXCEPT(rpc_filter)
263           {
264               status2 = GetExceptionCode();
265           }
266           __ENDTRY
267           if (status2 == RPC_S_SERVER_UNAVAILABLE &&
268               is_epm_destination_local(handle))
269           {
270               if (start_rpcss())
271                   continue;
272           }
273           if (status2 != RPC_S_OK)
274               ERR("ept_insert failed with error %d\n", status2);
275           status = status2; /* FIXME: convert status? */
276           break;
277       }
278   }
279   RpcBindingFree(&handle);
280 
281   for (i = 0; i < BindingVector->Count; i++)
282   {
283       unsigned j;
284       for (j = 0; j < (UuidVector ? UuidVector->Count : 1); j++)
285           I_RpcFree(entries[i*(UuidVector ? UuidVector->Count : 1) + j].tower);
286   }
287 
288   HeapFree(GetProcessHeap(), 0, entries);
289 
290   return status;
291 }
292 
293 /***********************************************************************
294  *             RpcEpRegisterA (RPCRT4.@)
295  */
296 RPC_STATUS WINAPI RpcEpRegisterA( RPC_IF_HANDLE IfSpec, RPC_BINDING_VECTOR *BindingVector,
297                                   UUID_VECTOR *UuidVector, RPC_CSTR Annotation )
298 {
299     return epm_register(IfSpec, BindingVector, UuidVector, Annotation, TRUE);
300 }
301 
302 /***********************************************************************
303  *             RpcEpRegisterNoReplaceA (RPCRT4.@)
304  */
305 RPC_STATUS WINAPI RpcEpRegisterNoReplaceA( RPC_IF_HANDLE IfSpec, RPC_BINDING_VECTOR *BindingVector,
306                                            UUID_VECTOR *UuidVector, RPC_CSTR Annotation )
307 {
308     return epm_register(IfSpec, BindingVector, UuidVector, Annotation, FALSE);
309 }
310 
311 /***********************************************************************
312  *             RpcEpRegisterW (RPCRT4.@)
313  */
314 RPC_STATUS WINAPI RpcEpRegisterW( RPC_IF_HANDLE IfSpec, RPC_BINDING_VECTOR *BindingVector,
315                                   UUID_VECTOR *UuidVector, RPC_WSTR Annotation )
316 {
317   LPSTR annA = RPCRT4_strdupWtoA(Annotation);
318   RPC_STATUS status;
319 
320   status = epm_register(IfSpec, BindingVector, UuidVector, (RPC_CSTR)annA, TRUE);
321 
322   HeapFree(GetProcessHeap(), 0, annA);
323   return status;
324 }
325 
326 /***********************************************************************
327  *             RpcEpRegisterNoReplaceW (RPCRT4.@)
328  */
329 RPC_STATUS WINAPI RpcEpRegisterNoReplaceW( RPC_IF_HANDLE IfSpec, RPC_BINDING_VECTOR *BindingVector,
330                                            UUID_VECTOR *UuidVector, RPC_WSTR Annotation )
331 {
332   LPSTR annA = RPCRT4_strdupWtoA(Annotation);
333   RPC_STATUS status;
334 
335   status = epm_register(IfSpec, BindingVector, UuidVector, (RPC_CSTR)annA, FALSE);
336 
337   HeapFree(GetProcessHeap(), 0, annA);
338   return status;
339 }
340 
341 /***********************************************************************
342  *             RpcEpUnregister (RPCRT4.@)
343  */
344 RPC_STATUS WINAPI RpcEpUnregister( RPC_IF_HANDLE IfSpec, RPC_BINDING_VECTOR *BindingVector,
345                                    UUID_VECTOR *UuidVector )
346 {
347   PRPC_SERVER_INTERFACE If = IfSpec;
348   ULONG i;
349   RPC_STATUS status = RPC_S_OK;
350   error_status_t status2;
351   ept_entry_t *entries;
352   handle_t handle;
353 
354   TRACE("(%p,%p,%p)\n", IfSpec, BindingVector, UuidVector);
355   TRACE(" ifid=%s\n", debugstr_guid(&If->InterfaceId.SyntaxGUID));
356   for (i=0; i<BindingVector->Count; i++) {
357     RpcBinding* bind = BindingVector->BindingH[i];
358     TRACE(" protseq[%d]=%s\n", i, debugstr_a(bind->Protseq));
359     TRACE(" endpoint[%d]=%s\n", i, debugstr_a(bind->Endpoint));
360   }
361   if (UuidVector) {
362     for (i=0; i<UuidVector->Count; i++)
363       TRACE(" obj[%d]=%s\n", i, debugstr_guid(UuidVector->Uuid[i]));
364   }
365 
366   entries = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*entries) * BindingVector->Count * (UuidVector ? UuidVector->Count : 1));
367   if (!entries)
368       return RPC_S_OUT_OF_MEMORY;
369 
370   status = get_epm_handle_server(&handle);
371   if (status != RPC_S_OK)
372   {
373     HeapFree(GetProcessHeap(), 0, entries);
374     return status;
375   }
376 
377   for (i = 0; i < BindingVector->Count; i++)
378   {
379       unsigned j;
380       RpcBinding* bind = BindingVector->BindingH[i];
381       for (j = 0; j < (UuidVector ? UuidVector->Count : 1); j++)
382       {
383           status = TowerConstruct(&If->InterfaceId, &If->TransferSyntax,
384                                   bind->Protseq, bind->Endpoint,
385                                   bind->NetworkAddr,
386                                   &entries[i*(UuidVector ? UuidVector->Count : 1) + j].tower);
387           if (status != RPC_S_OK) break;
388 
389           if (UuidVector)
390               memcpy(&entries[i * UuidVector->Count + j].object, &UuidVector->Uuid[j], sizeof(GUID));
391           else
392               memset(&entries[i].object, 0, sizeof(entries[i].object));
393       }
394   }
395 
396   if (status == RPC_S_OK)
397   {
398       __TRY
399       {
400           ept_insert(handle, BindingVector->Count * (UuidVector ? UuidVector->Count : 1),
401                      entries, TRUE, &status2);
402       }
403       __EXCEPT(rpc_filter)
404       {
405           status2 = GetExceptionCode();
406       }
407       __ENDTRY
408       if (status2 == RPC_S_SERVER_UNAVAILABLE)
409           status2 = EPT_S_NOT_REGISTERED;
410       if (status2 != RPC_S_OK)
411           ERR("ept_insert failed with error %d\n", status2);
412       status = status2; /* FIXME: convert status? */
413   }
414   RpcBindingFree(&handle);
415 
416   for (i = 0; i < BindingVector->Count; i++)
417   {
418       unsigned j;
419       for (j = 0; j < (UuidVector ? UuidVector->Count : 1); j++)
420           I_RpcFree(entries[i*(UuidVector ? UuidVector->Count : 1) + j].tower);
421   }
422 
423   HeapFree(GetProcessHeap(), 0, entries);
424 
425   return status;
426 }
427 
428 /***********************************************************************
429  *             RpcEpResolveBinding (RPCRT4.@)
430  */
431 RPC_STATUS WINAPI RpcEpResolveBinding( RPC_BINDING_HANDLE Binding, RPC_IF_HANDLE IfSpec )
432 {
433   PRPC_CLIENT_INTERFACE If = IfSpec;
434   RpcBinding* bind = Binding;
435   RPC_STATUS status;
436   error_status_t status2;
437   handle_t handle;
438   ept_lookup_handle_t entry_handle = NULL;
439   twr_t *tower;
440   twr_t *towers[4] = { NULL };
441   unsigned32 num_towers, i;
442   GUID uuid = GUID_NULL;
443   char *resolved_endpoint = NULL;
444 
445   TRACE("(%p,%p)\n", Binding, IfSpec);
446   TRACE(" protseq=%s\n", debugstr_a(bind->Protseq));
447   TRACE(" obj=%s\n", debugstr_guid(&bind->ObjectUuid));
448   TRACE(" networkaddr=%s\n", debugstr_a(bind->NetworkAddr));
449   TRACE(" ifid=%s\n", debugstr_guid(&If->InterfaceId.SyntaxGUID));
450 
451   /* just return for fully bound handles */
452   if (bind->Endpoint && (bind->Endpoint[0] != '\0'))
453     return RPC_S_OK;
454 
455   status = get_epm_handle_client(Binding, &handle);
456   if (status != RPC_S_OK) return status;
457 
458   status = TowerConstruct(&If->InterfaceId, &If->TransferSyntax, bind->Protseq,
459                           ((RpcBinding *)handle)->Endpoint,
460                           bind->NetworkAddr, &tower);
461   if (status != RPC_S_OK)
462   {
463       WARN("couldn't get tower\n");
464       RpcBindingFree(&handle);
465       return status;
466   }
467 
468   while (TRUE)
469   {
470     __TRY
471     {
472       ept_map(handle, &uuid, tower, &entry_handle, ARRAY_SIZE(towers), &num_towers, towers, &status2);
473       /* FIXME: translate status2? */
474     }
475     __EXCEPT(rpc_filter)
476     {
477       status2 = GetExceptionCode();
478     }
479     __ENDTRY
480     if (status2 == RPC_S_SERVER_UNAVAILABLE &&
481         is_epm_destination_local(handle))
482     {
483       if (start_rpcss())
484         continue;
485     }
486     break;
487   };
488 
489   RpcBindingFree(&handle);
490   I_RpcFree(tower);
491 
492   if (status2 != RPC_S_OK)
493   {
494     ERR("ept_map failed for ifid %s, protseq %s, networkaddr %s\n", debugstr_guid(&If->TransferSyntax.SyntaxGUID), bind->Protseq, bind->NetworkAddr);
495     return status2;
496   }
497 
498   for (i = 0; i < num_towers; i++)
499   {
500     /* only parse the tower if we haven't already found a suitable
501     * endpoint, otherwise just free the tower */
502     if (!resolved_endpoint)
503     {
504       status = TowerExplode(towers[i], NULL, NULL, NULL, &resolved_endpoint, NULL);
505       TRACE("status = %d\n", status);
506     }
507     I_RpcFree(towers[i]);
508   }
509 
510   if (resolved_endpoint)
511   {
512     RPCRT4_ResolveBinding(Binding, resolved_endpoint);
513     I_RpcFree(resolved_endpoint);
514     return RPC_S_OK;
515   }
516 
517   WARN("couldn't find an endpoint\n");
518   return EPT_S_NOT_REGISTERED;
519 }
520 
521 /*****************************************************************************
522  * TowerExplode (RPCRT4.@)
523  */
524 RPC_STATUS WINAPI TowerExplode(
525     const twr_t *tower, PRPC_SYNTAX_IDENTIFIER object, PRPC_SYNTAX_IDENTIFIER syntax,
526     char **protseq, char **endpoint, char **address)
527 {
528     size_t tower_size;
529     RPC_STATUS status;
530     const unsigned char *p;
531     u_int16 floor_count;
532     const twr_uuid_floor_t *object_floor;
533     const twr_uuid_floor_t *syntax_floor;
534 
535     TRACE("(%p, %p, %p, %p, %p, %p)\n", tower, object, syntax, protseq,
536           endpoint, address);
537 
538     if (protseq)
539         *protseq = NULL;
540     if (endpoint)
541         *endpoint = NULL;
542     if (address)
543         *address = NULL;
544 
545     tower_size = tower->tower_length;
546 
547     if (tower_size < sizeof(u_int16))
548         return EPT_S_NOT_REGISTERED;
549 
550     p = &tower->tower_octet_string[0];
551 
552     floor_count = *(const u_int16 *)p;
553     p += sizeof(u_int16);
554     tower_size -= sizeof(u_int16);
555     TRACE("floor_count: %d\n", floor_count);
556     /* FIXME: should we do something with the floor count? at the moment we don't */
557 
558     if (tower_size < sizeof(*object_floor) + sizeof(*syntax_floor))
559         return EPT_S_NOT_REGISTERED;
560 
561     object_floor = (const twr_uuid_floor_t *)p;
562     p += sizeof(*object_floor);
563     tower_size -= sizeof(*object_floor);
564     syntax_floor = (const twr_uuid_floor_t *)p;
565     p += sizeof(*syntax_floor);
566     tower_size -= sizeof(*syntax_floor);
567 
568     if ((object_floor->count_lhs != sizeof(object_floor->protid) +
569         sizeof(object_floor->uuid) + sizeof(object_floor->major_version)) ||
570         (object_floor->protid != EPM_PROTOCOL_UUID) ||
571         (object_floor->count_rhs != sizeof(object_floor->minor_version)))
572         return EPT_S_NOT_REGISTERED;
573 
574     if ((syntax_floor->count_lhs != sizeof(syntax_floor->protid) +
575         sizeof(syntax_floor->uuid) + sizeof(syntax_floor->major_version)) ||
576         (syntax_floor->protid != EPM_PROTOCOL_UUID) ||
577         (syntax_floor->count_rhs != sizeof(syntax_floor->minor_version)))
578         return EPT_S_NOT_REGISTERED;
579 
580     status = RpcTransport_ParseTopOfTower(p, tower_size, protseq, address, endpoint);
581     if ((status == RPC_S_OK) && syntax && object)
582     {
583         syntax->SyntaxGUID = syntax_floor->uuid;
584         syntax->SyntaxVersion.MajorVersion = syntax_floor->major_version;
585         syntax->SyntaxVersion.MinorVersion = syntax_floor->minor_version;
586         object->SyntaxGUID = object_floor->uuid;
587         object->SyntaxVersion.MajorVersion = object_floor->major_version;
588         object->SyntaxVersion.MinorVersion = object_floor->minor_version;
589     }
590     return status;
591 }
592 
593 /***********************************************************************
594  *             TowerConstruct (RPCRT4.@)
595  */
596 RPC_STATUS WINAPI TowerConstruct(
597     const RPC_SYNTAX_IDENTIFIER *object, const RPC_SYNTAX_IDENTIFIER *syntax,
598     const char *protseq, const char *endpoint, const char *address,
599     twr_t **tower)
600 {
601     size_t tower_size;
602     RPC_STATUS status;
603     unsigned char *p;
604     twr_uuid_floor_t *object_floor;
605     twr_uuid_floor_t *syntax_floor;
606 
607     TRACE("(%p, %p, %s, %s, %s, %p)\n", object, syntax, debugstr_a(protseq),
608           debugstr_a(endpoint), debugstr_a(address), tower);
609 
610     *tower = NULL;
611 
612     status = RpcTransport_GetTopOfTower(NULL, &tower_size, protseq, address, endpoint);
613 
614     if (status != RPC_S_OK)
615         return status;
616 
617     tower_size += sizeof(u_int16) + sizeof(*object_floor) + sizeof(*syntax_floor);
618     *tower = I_RpcAllocate(FIELD_OFFSET(twr_t, tower_octet_string[tower_size]));
619     if (!*tower)
620         return RPC_S_OUT_OF_RESOURCES;
621 
622     (*tower)->tower_length = tower_size;
623     p = &(*tower)->tower_octet_string[0];
624     *(u_int16 *)p = 5; /* number of floors */
625     p += sizeof(u_int16);
626     object_floor = (twr_uuid_floor_t *)p;
627     p += sizeof(*object_floor);
628     syntax_floor = (twr_uuid_floor_t *)p;
629     p += sizeof(*syntax_floor);
630 
631     object_floor->count_lhs = sizeof(object_floor->protid) + sizeof(object_floor->uuid) +
632                               sizeof(object_floor->major_version);
633     object_floor->protid = EPM_PROTOCOL_UUID;
634     object_floor->count_rhs = sizeof(object_floor->minor_version);
635     object_floor->uuid = object->SyntaxGUID;
636     object_floor->major_version = object->SyntaxVersion.MajorVersion;
637     object_floor->minor_version = object->SyntaxVersion.MinorVersion;
638 
639     syntax_floor->count_lhs = sizeof(syntax_floor->protid) + sizeof(syntax_floor->uuid) +
640                               sizeof(syntax_floor->major_version);
641     syntax_floor->protid = EPM_PROTOCOL_UUID;
642     syntax_floor->count_rhs = sizeof(syntax_floor->minor_version);
643     syntax_floor->uuid = syntax->SyntaxGUID;
644     syntax_floor->major_version = syntax->SyntaxVersion.MajorVersion;
645     syntax_floor->minor_version = syntax->SyntaxVersion.MinorVersion;
646 
647     status = RpcTransport_GetTopOfTower(p, &tower_size, protseq, address, endpoint);
648     if (status != RPC_S_OK)
649     {
650         I_RpcFree(*tower);
651         *tower = NULL;
652         return status;
653     }
654     return RPC_S_OK;
655 }
656 
657 void __RPC_FAR * __RPC_USER MIDL_user_allocate(SIZE_T len)
658 {
659     return HeapAlloc(GetProcessHeap(), 0, len);
660 }
661 
662 void __RPC_USER MIDL_user_free(void __RPC_FAR * ptr)
663 {
664     HeapFree(GetProcessHeap(), 0, ptr);
665 }
666