Added support for mappings with no associated file.

diff --git a/server/mapping.c b/server/mapping.c
index b6f4b68..acacbb6 100644
--- a/server/mapping.c
+++ b/server/mapping.c
@@ -9,6 +9,7 @@
 #include <stdlib.h>
 #include <unistd.h>
 
+#include "config.h"
 #include "winerror.h"
 #include "winnt.h"
 #include "server/process.h"
@@ -40,10 +41,51 @@
     mapping_destroy
 };
 
+#ifdef __i386__
+
+/* These are always the same on an i386, and it will be faster this way */
+# define page_mask  0xfff
+# define page_shift 12
+# define init_page_size() /* nothing */
+
+#else  /* __i386__ */
+
+static int page_shift, page_mask;
+
+static void init_page_size(void)
+{
+    int page_size;
+# ifdef HAVE_GETPAGESIZE
+    page_size = getpagesize();
+# else
+#  ifdef __svr4__
+    page_size = sysconf(_SC_PAGESIZE);
+#  else
+#   error Cannot get the page size on this platform
+#  endif
+# endif
+    page_mask = page_size - 1;
+    /* Make sure we have a power of 2 */
+    assert( !(page_size & page_mask) );
+    page_shift = 0;
+    while ((1 << page_shift) != page_size) page_shift++;
+}
+#endif  /* __i386__ */
+
+#define ROUND_ADDR(addr) \
+   ((int)(addr) & ~page_mask)
+
+#define ROUND_SIZE(addr,size) \
+   (((int)(size) + ((int)(addr) & page_mask) + page_mask) & ~page_mask)
+
+
 struct object *create_mapping( int size_high, int size_low, int protect,
                                int handle, const char *name )
 {
     struct mapping *mapping;
+    int access = 0;
+
+    if (!page_mask) init_page_size();
 
     if (!(mapping = (struct mapping *)create_named_object( name, &mapping_ops,
                                                            sizeof(*mapping) )))
@@ -51,22 +93,42 @@
     if (GET_ERROR() == ERROR_ALREADY_EXISTS)
         return &mapping->obj;  /* Nothing else to do */
 
+    if (protect & VPROT_READ) access |= GENERIC_READ;
+    if (protect & VPROT_WRITE) access |= GENERIC_WRITE;
+
+    size_low = ROUND_SIZE( 0, size_low );
+    if (handle != -1)
+    {
+        if (!(mapping->file = get_file_obj( current->process, handle, access ))) goto error;
+        if (!size_high && !size_low)
+        {
+            struct get_file_info_reply reply;
+            struct object *obj = (struct object *)mapping->file;
+            obj->ops->get_file_info( obj, &reply );
+            size_high = reply.size_high;
+            size_low  = ROUND_SIZE( 0, reply.size_low );
+        }
+        else if (!grow_file( mapping->file, size_high, size_low )) goto error;
+    }
+    else  /* Anonymous mapping (no associated file) */
+    {
+        if (!size_high && !size_low)
+        {
+            SET_ERROR( ERROR_INVALID_PARAMETER );
+            mapping->file = NULL;
+            goto error;
+        }
+        if (!(mapping->file = create_temp_file( access ))) goto error;
+        if (!grow_file( mapping->file, size_high, size_low )) goto error;
+    }
     mapping->size_high = size_high;
     mapping->size_low  = size_low;
     mapping->protect   = protect;
-    if (handle != -1)
-    {
-        int access = 0;
-        if (protect & VPROT_READ) access |= GENERIC_READ;
-        if (protect & VPROT_WRITE) access |= GENERIC_WRITE;
-        if (!(mapping->file = get_file_obj( current->process, handle, access )))
-        {
-            release_object( mapping );
-            return NULL;
-        }
-    }
-    else mapping->file = NULL;
     return &mapping->obj;
+
+ error:
+    release_object( mapping );
+    return NULL;
 }
 
 int open_mapping( unsigned int access, int inherit, const char *name )