Added handling of SBM_GETSCROLLBARINFO message.
[wine/wine-gecko.git] / dlls / rpcrt4 / rpcss_np_client.c
blob567277ee8687e5a12ea79480acfb34b39409ed7e
1 /*
2 * RPCSS named pipe client implementation
4 * Copyright (C) 2002 Greg Turner
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 #include <assert.h>
22 #include <stdarg.h>
24 #include "windef.h"
25 #include "winbase.h"
26 #include "ntstatus.h"
27 #include "wine/rpcss_shared.h"
28 #include "wine/debug.h"
30 #include "rpc_binding.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(ole);
34 HANDLE RPCRT4_RpcssNPConnect(void)
36 HANDLE the_pipe = NULL;
37 DWORD dwmode, wait_result;
38 HANDLE master_mutex = RPCRT4_GetMasterMutex();
40 TRACE("\n");
42 while (TRUE) {
44 wait_result = WaitForSingleObject(master_mutex, MASTER_MUTEX_TIMEOUT);
45 switch (wait_result) {
46 case WAIT_ABANDONED:
47 case WAIT_OBJECT_0:
48 break;
49 case WAIT_FAILED:
50 case WAIT_TIMEOUT:
51 default:
52 ERR("This should never happen: couldn't enter mutex.\n");
53 return NULL;
56 /* try to open the client side of the named pipe. */
57 the_pipe = CreateFileA(
58 NAME_RPCSS_NAMED_PIPE, /* pipe name */
59 GENERIC_READ | GENERIC_WRITE, /* r/w access */
60 0, /* no sharing */
61 NULL, /* no security attributes */
62 OPEN_EXISTING, /* open an existing pipe */
63 0, /* default attributes */
64 NULL /* no template file */
67 if (the_pipe != INVALID_HANDLE_VALUE)
68 break;
70 if (GetLastError() != ERROR_PIPE_BUSY) {
71 WARN("Unable to open named pipe %s (assuming unavailable).\n",
72 debugstr_a(NAME_RPCSS_NAMED_PIPE));
73 the_pipe = NULL;
74 break;
77 WARN("Named pipe busy (will wait)\n");
79 if (!ReleaseMutex(master_mutex))
80 ERR("Failed to release master mutex. Expect deadlock.\n");
82 /* wait for the named pipe. We are only
83 willing to wait only 5 seconds. It should be available /very/ soon. */
84 if (! WaitNamedPipeA(NAME_RPCSS_NAMED_PIPE, MASTER_MUTEX_WAITNAMEDPIPE_TIMEOUT))
86 ERR("Named pipe unavailable after waiting. Something is probably wrong.\n");
87 return NULL;
92 if (the_pipe) {
93 dwmode = PIPE_READMODE_MESSAGE;
94 /* SetNamedPipeHandleState not implemented ATM, but still seems to work somehow. */
95 if (! SetNamedPipeHandleState(the_pipe, &dwmode, NULL, NULL))
96 WARN("Failed to set pipe handle state\n");
99 if (!ReleaseMutex(master_mutex))
100 ERR("Uh oh, failed to leave the RPC Master Mutex!\n");
102 return the_pipe;
105 BOOL RPCRT4_SendReceiveNPMsg(HANDLE np, PRPCSS_NP_MESSAGE msg, char *vardata, PRPCSS_NP_REPLY reply)
107 DWORD count;
108 UINT32 payload_offset;
109 RPCSS_NP_MESSAGE vardata_payload_msg;
111 TRACE("(np == %p, msg == %p, vardata == %p, reply == %p)\n",
112 np, msg, vardata, reply);
114 if (! WriteFile(np, msg, sizeof(RPCSS_NP_MESSAGE), &count, NULL)) {
115 ERR("write failed.\n");
116 return FALSE;
119 if (count != sizeof(RPCSS_NP_MESSAGE)) {
120 ERR("write count mismatch.\n");
121 return FALSE;
124 /* process the vardata payload if necessary */
125 vardata_payload_msg.message_type = RPCSS_NP_MESSAGE_TYPEID_VARDATAPAYLOADMSG;
126 vardata_payload_msg.vardata_payload_size = 0; /* meaningless */
127 for ( payload_offset = 0; payload_offset < msg->vardata_payload_size;
128 payload_offset += VARDATA_PAYLOAD_BYTES ) {
129 TRACE("sending vardata payload. vd=%p, po=%d, ps=%d\n", vardata,
130 payload_offset, msg->vardata_payload_size);
131 ZeroMemory(vardata_payload_msg.message.vardatapayloadmsg.payload, VARDATA_PAYLOAD_BYTES);
132 CopyMemory(vardata_payload_msg.message.vardatapayloadmsg.payload,
133 vardata,
134 min( VARDATA_PAYLOAD_BYTES, msg->vardata_payload_size - payload_offset ));
135 vardata += VARDATA_PAYLOAD_BYTES;
136 if (! WriteFile(np, &vardata_payload_msg, sizeof(RPCSS_NP_MESSAGE), &count, NULL)) {
137 ERR("vardata write failed at %u bytes.\n", payload_offset);
138 return FALSE;
142 if (! ReadFile(np, reply, sizeof(RPCSS_NP_REPLY), &count, NULL)) {
143 ERR("read failed.\n");
144 return FALSE;
147 if (count != sizeof(RPCSS_NP_REPLY)) {
148 ERR("read count mismatch. got %ld, expected %u.\n", count, sizeof(RPCSS_NP_REPLY));
149 return FALSE;
152 /* message execution was successful */
153 return TRUE;