Implement mailslots.
diff --git a/dlls/kernel/file.c b/dlls/kernel/file.c
index 8ebdd22..5242885 100644
--- a/dlls/kernel/file.c
+++ b/dlls/kernel/file.c
@@ -1217,9 +1217,11 @@
if (!strncmpW(filename, bkslashes_with_dotW, 4))
{
static const WCHAR pipeW[] = {'P','I','P','E','\\',0};
+ static const WCHAR mailslotW[] = {'M','A','I','L','S','L','O','T','\\',0};
if ((isalphaW(filename[4]) && filename[5] == ':' && filename[6] == '\0') ||
- !strncmpiW( filename + 4, pipeW, 5 ))
+ !strncmpiW( filename + 4, pipeW, 5 ) ||
+ !strncmpiW( filename + 4, mailslotW, 9 ))
{
dosdev = 0;
}
diff --git a/dlls/kernel/sync.c b/dlls/kernel/sync.c
index 4e4d485..bd4cc72 100644
--- a/dlls/kernel/sync.c
+++ b/dlls/kernel/sync.c
@@ -1641,9 +1641,31 @@
LPDWORD lpNextSize, LPDWORD lpMessageCount,
LPDWORD lpReadTimeout )
{
- FIXME("(%p): stub\n",hMailslot);
- SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
- return FALSE;
+ BOOL r;
+
+ TRACE("%p %p %p %p %p\n",hMailslot,
+ lpMaxMessageSize,lpNextSize,lpMessageCount,lpReadTimeout);
+
+ SERVER_START_REQ( set_mailslot_info )
+ {
+ req->handle = hMailslot;
+ req->flags = 0;
+ r = !wine_server_call_err( req );
+ if( r )
+ {
+ if( lpMaxMessageSize )
+ *lpMaxMessageSize = reply->max_msgsize;
+ if( lpNextSize )
+ *lpNextSize = reply->next_msgsize;
+ if( lpMessageCount )
+ *lpMessageCount = reply->msg_count;
+ if( lpReadTimeout )
+ *lpReadTimeout = reply->read_timeout;
+ }
+ }
+ SERVER_END_REQ;
+
+ return r;
}
@@ -1662,9 +1684,20 @@
*/
BOOL WINAPI SetMailslotInfo( HANDLE hMailslot, DWORD dwReadTimeout)
{
- FIXME("%p %ld: stub\n", hMailslot, dwReadTimeout);
- SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
- return FALSE;
+ BOOL r;
+
+ TRACE("%p %ld\n", hMailslot, dwReadTimeout);
+
+ SERVER_START_REQ( set_mailslot_info )
+ {
+ req->handle = hMailslot;
+ req->flags = MAILSLOT_SET_READ_TIMEOUT;
+ req->read_timeout = dwReadTimeout;
+ r = !wine_server_call_err( req );
+ }
+ SERVER_END_REQ;
+
+ return r;
}
diff --git a/dlls/kernel/tests/mailslot.c b/dlls/kernel/tests/mailslot.c
index 989ec53..735fea8 100644
--- a/dlls/kernel/tests/mailslot.c
+++ b/dlls/kernel/tests/mailslot.c
@@ -67,8 +67,6 @@
ok( GetLastError() == ERROR_PATH_NOT_FOUND,
"error should be ERROR_PATH_NOT_FOUND\n");
- todo_wine
- {
/* valid open, but with wacky parameters ... then check them */
hSlot = CreateMailslot( szmspath, -1, -1, NULL );
ok( hSlot != INVALID_HANDLE_VALUE , "mailslot with valid name failed\n");
@@ -77,22 +75,15 @@
"getmailslotinfo failed\n");
ok( dwMax == ~0UL, "dwMax incorrect\n");
ok( dwNext == MAILSLOT_NO_MESSAGE, "dwNext incorrect\n");
- }
ok( dwMsgCount == 0, "dwMsgCount incorrect\n");
- todo_wine
- {
ok( dwTimeout == ~0UL, "dwTimeout incorrect\n");
ok( GetMailslotInfo( hSlot, NULL, NULL, NULL, NULL ),
"getmailslotinfo failed\n");
ok( CloseHandle(hSlot), "failed to close mailslot\n");
- }
- todo_wine
- {
/* now open it for real */
hSlot = CreateMailslot( szmspath, 0, 0, NULL );
ok( hSlot != INVALID_HANDLE_VALUE , "valid mailslot failed\n");
- }
/* try and read/write to it */
count = 0;
@@ -106,8 +97,6 @@
hWriter = CreateFile(szmspath, GENERIC_READ|GENERIC_WRITE,
0, NULL, OPEN_EXISTING, 0, NULL);
ok( hWriter == INVALID_HANDLE_VALUE, "bad sharing mode\n");
- todo_wine
- {
ok( GetLastError() == ERROR_SHARING_VIOLATION,
"error should be ERROR_SHARING_VIOLATION\n");
@@ -115,7 +104,6 @@
hWriter = CreateFile(szmspath, GENERIC_READ|GENERIC_WRITE,
FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
ok( hWriter != INVALID_HANDLE_VALUE, "existing mailslot\n");
- }
/*
* opening a client should make no difference to
@@ -132,11 +120,8 @@
*/
ok( !ReadFile( hWriter, buffer, sizeof buffer/2, &count, NULL),
"can read client\n");
- todo_wine
- {
ok( WriteFile( hWriter, buffer, sizeof buffer/2, &count, NULL),
"can't write client\n");
- }
ok( !ReadFile( hWriter, buffer, sizeof buffer/2, &count, NULL),
"can read client\n");
@@ -144,12 +129,9 @@
* seeing as there's something in the slot,
* we should be able to read it once
*/
- todo_wine
- {
ok( ReadFile( hSlot, buffer, sizeof buffer, &count, NULL),
"slot read\n");
ok( count == (sizeof buffer/2), "short read\n" );
- }
/* but not again */
ok( !ReadFile( hSlot, buffer, sizeof buffer, &count, NULL),
@@ -174,8 +156,6 @@
hSlot2 = CreateMailslot( szmspath, 0, 0, NULL );
ok( hSlot2 == INVALID_HANDLE_VALUE , "opened two mailslots\n");
- todo_wine
- {
/* close the client again */
ok( CloseHandle( hWriter ), "closing the client\n");
@@ -186,7 +166,6 @@
hWriter = CreateFile(szmspath, GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
ok( hWriter != INVALID_HANDLE_VALUE, "sharing writer\n");
- }
/*
* now try open another as a writer ...
@@ -196,8 +175,6 @@
FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
ok( hWriter2 == INVALID_HANDLE_VALUE, "greedy writer succeeded\n");
- todo_wine
- {
/* now try open another as a writer ... and share with the first */
hWriter2 = CreateFile(szmspath, GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
@@ -208,7 +185,6 @@
ok( GetMailslotInfo( hSlot, &dwMax, &dwNext, &dwMsgCount, &dwTimeout ),
"getmailslotinfo failed\n");
ok( dwNext == MAILSLOT_NO_MESSAGE, "dwNext incorrect\n");
- }
ok( dwMax == 0, "dwMax incorrect\n");
ok( dwMsgCount == 0, "dwMsgCount incorrect\n");
ok( dwTimeout == 0, "dwTimeout incorrect\n");
@@ -217,8 +193,6 @@
ok( !ReadFile( hSlot, buffer, sizeof buffer, &count, NULL), "slot read\n");
/* write two messages */
- todo_wine
- {
buffer[0] = 'a';
ok( WriteFile( hWriter, buffer, 1, &count, NULL), "1st write failed\n");
@@ -226,7 +200,9 @@
dwNext = dwMsgCount = 0;
ok( GetMailslotInfo( hSlot, NULL, &dwNext, &dwMsgCount, NULL ),
"getmailslotinfo failed\n");
+ todo_wine {
ok( dwNext == 1, "dwNext incorrect\n");
+ }
ok( dwMsgCount == 1, "dwMsgCount incorrect\n");
buffer[0] = 'b';
@@ -237,8 +213,10 @@
dwNext = dwMsgCount = 0;
ok( GetMailslotInfo( hSlot, NULL, &dwNext, &dwMsgCount, NULL ),
"getmailslotinfo failed\n");
+ todo_wine {
ok( dwNext == 1, "dwNext incorrect\n");
ok( dwMsgCount == 2, "dwMsgCount incorrect\n");
+ }
/* write a 3rd message with zero size */
ok( WriteFile( hWriter2, buffer, 0, &count, NULL), "3rd write failed\n");
@@ -247,8 +225,10 @@
dwNext = dwMsgCount = 0;
ok( GetMailslotInfo( hSlot, NULL, &dwNext, &dwMsgCount, NULL ),
"getmailslotinfo failed\n");
+ todo_wine {
ok( dwNext == 1, "dwNext incorrect\n");
ok( dwMsgCount == 3, "dwMsgCount incorrect\n");
+ }
buffer[0]=buffer[1]=0;
@@ -265,8 +245,10 @@
dwNext = dwMsgCount = 0;
ok( GetMailslotInfo( hSlot, NULL, &dwNext, &dwMsgCount, NULL ),
"getmailslotinfo failed\n");
+ todo_wine {
ok( dwNext == 2, "dwNext incorrect\n");
ok( dwMsgCount == 2, "dwMsgCount incorrect\n");
+ }
/* read the second message */
ok( ReadFile( hSlot, buffer, sizeof buffer, &count, NULL),
@@ -278,13 +260,13 @@
dwNext = dwMsgCount = 0;
ok( GetMailslotInfo( hSlot, NULL, &dwNext, &dwMsgCount, NULL ),
"getmailslotinfo failed\n");
- }
+ todo_wine {
ok( dwNext == 0, "dwNext incorrect\n");
- todo_wine
- {
ok( dwMsgCount == 1, "dwMsgCount incorrect\n");
+ }
/* read the 3rd (zero length) message */
+ todo_wine {
ok( ReadFile( hSlot, buffer, sizeof buffer, &count, NULL),
"3rd slot read failed\n");
}
@@ -294,13 +276,10 @@
* now there should be no more messages
* check the mailslot info
*/
- todo_wine
- {
dwNext = dwMsgCount = 0;
ok( GetMailslotInfo( hSlot, NULL, &dwNext, &dwMsgCount, NULL ),
"getmailslotinfo failed\n");
ok( dwNext == MAILSLOT_NO_MESSAGE, "dwNext incorrect\n");
- }
ok( dwMsgCount == 0, "dwMsgCount incorrect\n");
/* check that reads fail */
@@ -308,12 +287,9 @@
"3rd slot read succeeded\n");
/* finally close the mailslot and its client */
- todo_wine
- {
ok( CloseHandle( hWriter2 ), "closing 2nd client\n");
ok( CloseHandle( hWriter ), "closing the client\n");
ok( CloseHandle( hSlot ), "closing the mailslot\n");
- }
return 0;
}
diff --git a/dlls/ntdll/file.c b/dlls/ntdll/file.c
index 30c3cbd..92c1bf6 100644
--- a/dlls/ntdll/file.c
+++ b/dlls/ntdll/file.c
@@ -130,6 +130,7 @@
ULONG options, PVOID ea_buffer, ULONG ea_length )
{
static const WCHAR pipeW[] = {'\\','?','?','\\','p','i','p','e','\\'};
+ static const WCHAR mailslotW[] = {'\\','?','?','\\','M','A','I','L','S','L','O','T','\\'};
ANSI_STRING unix_name;
int created = FALSE;
@@ -166,6 +167,25 @@
return io->u.Status;
}
+ /* check for mailslot */
+
+ if (attr->ObjectName->Length > sizeof(mailslotW) &&
+ !memicmpW( attr->ObjectName->Buffer, mailslotW, sizeof(mailslotW)/sizeof(WCHAR) ))
+ {
+ SERVER_START_REQ( open_mailslot )
+ {
+ req->access = access & GENERIC_WRITE;
+ req->sharing = sharing;
+ req->inherit = (attr->Attributes & OBJ_INHERIT) != 0;
+ wine_server_add_data( req, attr->ObjectName->Buffer + 4,
+ attr->ObjectName->Length - 4*sizeof(WCHAR) );
+ io->u.Status = wine_server_call( req );
+ *handle = reply->handle;
+ }
+ SERVER_END_REQ;
+ return io->u.Status;
+ }
+
io->u.Status = wine_nt_to_unix_file_name( attr->ObjectName, &unix_name, disposition,
!(attr->Attributes & OBJ_CASE_INSENSITIVE) );
@@ -1766,7 +1786,7 @@
'\\','?','?','\\','M','A','I','L','S','L','O','T','\\'};
NTSTATUS ret;
- FIXME("%p %08lx %p %p %08lx %08lx %08lx %p\n",
+ TRACE("%p %08lx %p %p %08lx %08lx %08lx %p\n",
pHandle, DesiredAccess, attr, IoStatusBlock,
CreateOptions, MailslotQuota, MaxMessageSize, TimeOut);
@@ -1777,7 +1797,18 @@
return STATUS_OBJECT_NAME_INVALID;
}
- ret = STATUS_NOT_IMPLEMENTED;
-
+ SERVER_START_REQ( create_mailslot )
+ {
+ req->max_msgsize = MaxMessageSize;
+ req->read_timeout = TimeOut->QuadPart / -10000;
+ req->inherit = (attr->Attributes & OBJ_INHERIT) != 0;
+ wine_server_add_data( req, attr->ObjectName->Buffer + 4,
+ attr->ObjectName->Length - 4*sizeof(WCHAR) );
+ ret = wine_server_call( req );
+ if( ret == STATUS_SUCCESS )
+ *pHandle = reply->handle;
+ }
+ SERVER_END_REQ;
+
return ret;
}