2002-10-22 00:41:17 +00:00
|
|
|
/*
|
|
|
|
* NDR data marshalling
|
|
|
|
*
|
|
|
|
* Copyright 2002 Greg Turner
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
* TODO:
|
|
|
|
* - figure out whether we *really* got this right
|
|
|
|
* - check for errors and throw exceptions
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2002-10-25 19:03:43 +00:00
|
|
|
#include <assert.h>
|
2002-10-22 00:41:17 +00:00
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "winerror.h"
|
|
|
|
#include "winreg.h"
|
|
|
|
|
2002-10-25 19:03:43 +00:00
|
|
|
#include "ndr_misc.h"
|
|
|
|
|
|
|
|
#include "wine/rpcfc.h"
|
2002-10-22 00:41:17 +00:00
|
|
|
#include "wine/obj_base.h"
|
|
|
|
|
|
|
|
#include "wine/debug.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(ole);
|
|
|
|
|
2002-10-28 20:40:59 +00:00
|
|
|
#define BUFFER_PARANOIA 20
|
2002-10-25 19:03:43 +00:00
|
|
|
|
2002-10-28 18:47:41 +00:00
|
|
|
#if defined(__i386__)
|
|
|
|
#define LITTLE_ENDIAN_32_WRITE(pchar, word32) \
|
|
|
|
(*((UINT32 *)(pchar)) = (word32))
|
|
|
|
|
|
|
|
#define LITTLE_ENDIAN_32_READ(pchar) \
|
|
|
|
(*((UINT32 *)(pchar)))
|
|
|
|
#else
|
|
|
|
/* these would work for i386 too, but less efficient */
|
|
|
|
#define LITTLE_ENDIAN_32_WRITE(pchar, word32) \
|
|
|
|
(*(pchar) = LOBYTE(LOWORD(word32)), \
|
|
|
|
*((pchar)+1) = HIBYTE(LOWORD(word32)), \
|
|
|
|
*((pchar)+2) = LOBYTE(HIWORD(word32)), \
|
|
|
|
*((pchar)+3) = HIBYTE(HIWORD(word32)), \
|
|
|
|
(word32))
|
|
|
|
|
|
|
|
#define LITTLE_ENDIAN_32_READ(pchar) \
|
|
|
|
(MAKELONG( \
|
|
|
|
MAKEWORD(*(pchar), *((pchar)+1)) \
|
|
|
|
MAKEWORD(*((pchar)+2), *((pchar)+3)))
|
|
|
|
#endif
|
|
|
|
|
2002-10-22 00:41:17 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* NdrConformantStringMarshall [RPCRT4.@]
|
|
|
|
*/
|
|
|
|
unsigned char *WINAPI NdrConformantStringMarshall(MIDL_STUB_MESSAGE *pStubMsg, unsigned char *pszMessage,
|
|
|
|
PFORMAT_STRING pFormat)
|
|
|
|
{
|
2002-10-25 19:03:43 +00:00
|
|
|
UINT32 len, i;
|
|
|
|
unsigned char *c;
|
|
|
|
|
|
|
|
TRACE("(pStubMsg == ^%p, pszMessage == ^%p, pFormat == ^%p)\n", pStubMsg, pszMessage, pFormat);
|
|
|
|
|
2002-10-28 23:53:23 +00:00
|
|
|
assert(pFormat);
|
2002-10-25 19:03:43 +00:00
|
|
|
if (*pFormat == RPC_FC_C_CSTRING) {
|
|
|
|
len = strlen(pszMessage);
|
|
|
|
assert( (pStubMsg->BufferLength > (len + 13)) && (pStubMsg->Buffer != NULL) );
|
|
|
|
/* in DCE terminology this is a Conformant Varying String */
|
|
|
|
c = pStubMsg->Buffer;
|
2002-10-25 21:02:02 +00:00
|
|
|
memset(c, 0, 12);
|
2002-10-28 18:47:41 +00:00
|
|
|
LITTLE_ENDIAN_32_WRITE(c, len + 1); /* max length: strlen + 1 (for '\0') */
|
|
|
|
c += 8; /* offset: 0 */
|
|
|
|
LITTLE_ENDIAN_32_WRITE(c, len + 1); /* actual length: (same) */
|
2002-10-25 19:03:43 +00:00
|
|
|
c += 4;
|
|
|
|
for (i = 0; i <= len; i++)
|
|
|
|
*(c++) = *(pszMessage++); /* copy the string itself into the remaining space */
|
|
|
|
} else {
|
|
|
|
ERR("Unhandled string type: %#x\n", *pFormat);
|
2002-10-28 23:53:23 +00:00
|
|
|
/* FIXME: raise an exception. */
|
2002-10-25 19:03:43 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* success */
|
|
|
|
pStubMsg->fBufferValid = 1;
|
|
|
|
return NULL; /* is this always right? */
|
2002-10-22 00:41:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* NdrConformantStringBufferSize [RPCRT4.@]
|
|
|
|
*/
|
|
|
|
void WINAPI NdrConformantStringBufferSize(PMIDL_STUB_MESSAGE pStubMsg, unsigned char* pMemory, PFORMAT_STRING pFormat)
|
|
|
|
{
|
2002-10-25 19:03:43 +00:00
|
|
|
TRACE("(pStubMsg == ^%p, pMemory == ^%p, pFormat == ^%p)\n", pStubMsg, pMemory, pFormat);
|
|
|
|
|
2002-10-28 23:53:23 +00:00
|
|
|
assert(pFormat);
|
2002-10-25 19:03:43 +00:00
|
|
|
if (*pFormat == RPC_FC_C_CSTRING) {
|
2002-10-28 21:14:16 +00:00
|
|
|
/* we need 12 octets for the [maxlen, offset, len] DWORDS, + 1 octet for '\0' */
|
2002-10-28 20:40:59 +00:00
|
|
|
pStubMsg->BufferLength = strlen(pMemory) + 13 + BUFFER_PARANOIA;
|
2002-10-25 19:03:43 +00:00
|
|
|
} else {
|
|
|
|
ERR("Unhandled string type: %#x\n", *pFormat);
|
2002-10-28 23:53:23 +00:00
|
|
|
/* FIXME: raise an exception */
|
2002-10-25 19:03:43 +00:00
|
|
|
}
|
2002-10-22 00:41:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* NdrConformantStringMemorySize [RPCRT4.@]
|
|
|
|
*/
|
|
|
|
unsigned long WINAPI NdrConformantStringMemorySize( PMIDL_STUB_MESSAGE pStubMsg, PFORMAT_STRING pFormat )
|
|
|
|
{
|
2002-10-25 19:03:43 +00:00
|
|
|
FIXME("(pStubMsg == ^%p, pFormat == ^%p): stub.\n", pStubMsg, pFormat);
|
2002-10-22 00:41:17 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* NdrConformantStringUnmarshall [RPCRT4.@]
|
|
|
|
*/
|
|
|
|
unsigned char *WINAPI NdrConformantStringUnmarshall( PMIDL_STUB_MESSAGE pStubMsg, unsigned char** ppMemory,
|
|
|
|
PFORMAT_STRING pFormat, unsigned char fMustAlloc )
|
|
|
|
{
|
2002-10-28 23:53:23 +00:00
|
|
|
UINT32 len;
|
|
|
|
|
|
|
|
TRACE("(pStubMsg == ^%p, *pMemory == ^%p, pFormat == ^%p, fMustAlloc == %u)\n",
|
2002-10-28 20:07:01 +00:00
|
|
|
pStubMsg, *ppMemory, pFormat, fMustAlloc);
|
2002-10-28 23:53:23 +00:00
|
|
|
|
|
|
|
assert(pFormat && ppMemory && pStubMsg);
|
|
|
|
|
|
|
|
/* get the length and check that we handle the format string...
|
|
|
|
FIXME: this is probably what NdrConformantStringMemorySize is for... so move code there? */
|
|
|
|
|
|
|
|
if (*pFormat == RPC_FC_C_CSTRING) {
|
|
|
|
/* What we should have here is a varying conformant string (of single-octet-chars):
|
|
|
|
* [
|
|
|
|
* maxlen: DWORD
|
|
|
|
* offset: DWORD
|
|
|
|
* length: DWORD
|
|
|
|
* [
|
|
|
|
* data: char[], null terminated.
|
|
|
|
* ]
|
|
|
|
* ]
|
|
|
|
* I think what we want to do is advance the buffer pointer in the pStubMsg
|
|
|
|
* up to the point of the char after the last marshalled char.
|
|
|
|
* FIXME: it is probably supposed to really advance either length or maxlength
|
|
|
|
* chars... but which? For now I have ignored the issue and just positioned
|
|
|
|
* after the terminating '\0'; this is (probably) wrong and will (probably)
|
|
|
|
* cause problems until its fixed.
|
|
|
|
*/
|
|
|
|
|
|
|
|
pStubMsg->Buffer += 8; /* FIXME: do we care about maxlen? */
|
|
|
|
len = LITTLE_ENDIAN_32_READ(pStubMsg->Buffer);
|
|
|
|
pStubMsg->Buffer += 4;
|
|
|
|
} else {
|
|
|
|
ERR("Unhandled string type: %#x\n", *pFormat);
|
|
|
|
/* FIXME: raise an exception. */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* now the actual length (in bytes) that we need to store
|
|
|
|
the unmarshalled string is in len, including terminating '\0' */
|
|
|
|
|
|
|
|
if ( fMustAlloc || (!(*ppMemory)) || (pStubMsg->Memory != *ppMemory) ||
|
|
|
|
(pStubMsg->MemorySize < (len+BUFFER_PARANOIA)) ) {
|
|
|
|
/* crap, looks like we need to do something about the Memory. I don't
|
|
|
|
understand, it doesn't look like Microsoft is doing this the same
|
|
|
|
way... but then how do they do it? AFAICS the Memory is never deallocated by
|
|
|
|
the stub code so where does it go?... anyhow, I guess we'll just do it
|
|
|
|
our own way for now... */
|
|
|
|
pStubMsg->MemorySize = len;
|
|
|
|
pStubMsg->Memory = *ppMemory;
|
|
|
|
/* FIXME: pfnAllocate? or does that not apply to these "Memory" parts? */
|
|
|
|
*ppMemory = pStubMsg->Memory = HeapReAlloc(GetProcessHeap(), 0, pStubMsg->Memory, pStubMsg->MemorySize);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(pStubMsg->Memory)) {
|
|
|
|
ERR("Memory Allocation Failure\n");
|
|
|
|
/* FIXME: raise an exception */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* OK, we've got our ram. now do the real unmarshalling */
|
|
|
|
if (*pFormat == RPC_FC_C_CSTRING) {
|
|
|
|
char *c = *ppMemory;
|
|
|
|
|
|
|
|
while (*c != '\0')
|
|
|
|
*c++ = *(pStubMsg->Buffer++);
|
|
|
|
pStubMsg->Buffer++; /* advance past the '\0' */
|
|
|
|
|
|
|
|
} else
|
|
|
|
assert(FALSE);
|
|
|
|
|
|
|
|
return NULL; /*is this always right? */
|
2002-10-22 00:41:17 +00:00
|
|
|
}
|
2002-10-25 19:03:43 +00:00
|
|
|
|
2002-10-28 20:07:01 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* NdrConvert [RPCRT4.@]
|
|
|
|
*/
|
|
|
|
void WINAPI NdrConvert( PMIDL_STUB_MESSAGE pStubMsg, PFORMAT_STRING pFormat )
|
|
|
|
{
|
|
|
|
FIXME("(pStubMsg == ^%p, pFormat == ^%p): stub.\n", pStubMsg, pFormat);
|
2002-10-28 21:14:16 +00:00
|
|
|
/* FIXME: since this stub doesn't do any converting, the proper behavior
|
|
|
|
is to raise an exception */
|
2002-10-28 20:07:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* NdrConvert2 [RPCRT4.@]
|
|
|
|
*/
|
|
|
|
void WINAPI NdrConvert2( PMIDL_STUB_MESSAGE pStubMsg, PFORMAT_STRING pFormat, long NumberParams )
|
|
|
|
{
|
|
|
|
FIXME("(pStubMsg == ^%p, pFormat == ^%p, NumberParams == %ld): stub.\n", pStubMsg, pFormat, NumberParams);
|
2002-10-28 21:14:16 +00:00
|
|
|
/* FIXME: since this stub doesn't do any converting, the proper behavior
|
|
|
|
is to raise an exception */
|
2002-10-28 20:07:01 +00:00
|
|
|
}
|