10l, patch by Jan Knutar <jknutar@nic.fi>
[mplayer.git] / loader / ext.c
blobe7c68ce071dff00accc485110e1cdc9860f9c012
1 /********************************************************
4 * Stub functions for Wine module
7 ********************************************************/
8 #include "config.h"
9 #include <stdio.h>
10 #include <stdlib.h>
11 #ifdef HAVE_MALLOC_H
12 #include <malloc.h>
13 #endif
14 #include <unistd.h>
15 #include <sys/mman.h>
16 #include <errno.h>
17 #include <fcntl.h>
18 #include <string.h>
19 #include <stdarg.h>
20 #include <ctype.h>
22 #include "wine/windef.h"
23 #include "wine/winbase.h"
24 #include "wine/debugtools.h"
25 #include "wine/heap.h"
26 #include "ext.h"
28 #if 0
29 //REMOVE SIMPLIFY
30 static void* mymalloc(unsigned int size)
32 printf("malloc %d\n", size);
33 return malloc(size);
36 #undef malloc
37 #define malloc mymalloc
38 #endif
40 int dbg_header_err( const char *dbg_channel, const char *func )
42 return 0;
44 int dbg_header_warn( const char *dbg_channel, const char *func )
46 return 0;
48 int dbg_header_fixme( const char *dbg_channel, const char *func )
50 return 0;
52 int dbg_header_trace( const char *dbg_channel, const char *func )
54 return 0;
56 int dbg_vprintf( const char *format, va_list args )
58 return 0;
60 int __vprintf( const char *format, ... )
62 #ifdef DETAILED_OUT
63 va_list va;
64 va_start(va, format);
65 vprintf(format, va);
66 va_end(va);
67 #endif
68 return 0;
71 HANDLE WINAPI GetProcessHeap(void)
73 return 1;
76 LPVOID WINAPI HeapAlloc(HANDLE heap, DWORD flags, DWORD size)
78 static int i = 5;
79 void* m = (flags & 0x8) ? calloc(size, 1) : malloc(size);
80 //printf("HeapAlloc %p %d (%d)\n", m, size, flags);
81 //if (--i == 0)
82 // abort();
83 return m;
86 WIN_BOOL WINAPI HeapFree(HANDLE heap, DWORD flags, LPVOID mem)
88 if (mem) free(mem);
89 //printf("HeapFree %p\n", mem);
90 //if (!mem)
91 // abort();
92 return 1;
95 static int last_error;
97 DWORD WINAPI GetLastError(void)
99 return last_error;
102 VOID WINAPI SetLastError(DWORD error)
104 last_error=error;
107 WIN_BOOL WINAPI ReadFile(HANDLE handle, LPVOID mem, DWORD size, LPDWORD result, LPOVERLAPPED flags)
109 *result=read(handle, mem, size);
110 return *result;
112 INT WINAPI lstrcmpiA(LPCSTR c1, LPCSTR c2)
114 return strcasecmp(c1,c2);
116 LPSTR WINAPI lstrcpynA(LPSTR dest, LPCSTR src, INT num)
118 return strncpy(dest,src,num);
120 INT WINAPI lstrlenA(LPCSTR s)
122 return strlen(s);
124 INT WINAPI lstrlenW(LPCWSTR s)
126 int l;
127 if(!s)
128 return 0;
129 l=0;
130 while(s[l])
131 l++;
132 return l;
134 LPSTR WINAPI lstrcpynWtoA(LPSTR dest, LPCWSTR src, INT count)
136 LPSTR result = dest;
137 int moved=0;
138 if((dest==0) || (src==0))
139 return 0;
140 while(moved<count)
142 *dest=*src;
143 moved++;
144 if(*src==0)
145 break;
146 src++;
147 dest++;
149 return result;
151 /* i stands here for ignore case! */
152 int wcsnicmp(const unsigned short* s1, const unsigned short* s2, int n)
155 if(s1==0)
156 return;
157 if(s2==0)
158 return;
160 while(n>0)
162 if (((*s1 | *s2) & 0xff00) || toupper((char)*s1) != toupper((char)*s2))
165 if(*s1<*s2)
166 return -1;
167 else
168 if(*s1>*s2)
169 return 1;
170 else
171 if(*s1==0)
172 return 0;
174 s1++;
175 s2++;
176 n--;
178 return 0;
181 WIN_BOOL WINAPI IsBadReadPtr(LPCVOID data, UINT size)
183 if(size==0)
184 return 0;
185 if(data==NULL)
186 return 1;
187 return 0;
189 LPSTR HEAP_strdupA(HANDLE heap, DWORD flags, LPCSTR string)
191 // return strdup(string);
192 char* answ = (char*) malloc(strlen(string) + 1);
193 strcpy(answ, string);
194 return answ;
196 LPWSTR HEAP_strdupAtoW(HANDLE heap, DWORD flags, LPCSTR string)
198 int size, i;
199 WCHAR* answer;
200 if(string==0)
201 return 0;
202 size=strlen(string);
203 answer = (WCHAR*) malloc(sizeof(WCHAR) * (size + 1));
204 for(i=0; i<=size; i++)
205 answer[i]=(short)string[i];
206 return answer;
208 LPSTR HEAP_strdupWtoA(HANDLE heap, DWORD flags, LPCWSTR string)
210 int size, i;
211 char* answer;
212 if(string==0)
213 return 0;
214 size=0;
215 while(string[size])
216 size++;
217 answer = (char*) malloc(size + 2);
218 for(i=0; i<=size; i++)
219 answer[i]=(char)string[i];
220 return answer;
223 /***********************************************************************
224 * FILE_dommap
227 //#define MAP_PRIVATE
228 //#define MAP_SHARED
229 #undef MAP_ANON
230 LPVOID FILE_dommap( int unix_handle, LPVOID start,
231 DWORD size_high, DWORD size_low,
232 DWORD offset_high, DWORD offset_low,
233 int prot, int flags )
235 int fd = -1;
236 int pos;
237 LPVOID ret;
239 if (size_high || offset_high)
240 printf("offsets larger than 4Gb not supported\n");
242 if (unix_handle == -1)
244 #ifdef MAP_ANON
245 // printf("Anonymous\n");
246 flags |= MAP_ANON;
247 #else
248 static int fdzero = -1;
250 if (fdzero == -1)
252 if ((fdzero = open( "/dev/zero", O_RDONLY )) == -1)
254 perror( "Cannot open /dev/zero for READ. Check permissions! error: " );
255 exit(1);
258 fd = fdzero;
259 #endif /* MAP_ANON */
260 /* Linux EINVAL's on us if we don't pass MAP_PRIVATE to an anon mmap */
261 #ifdef MAP_SHARED
262 flags &= ~MAP_SHARED;
263 #endif
264 #ifdef MAP_PRIVATE
265 flags |= MAP_PRIVATE;
266 #endif
268 else fd = unix_handle;
269 // printf("fd %x, start %x, size %x, pos %x, prot %x\n",fd,start,size_low, offset_low, prot);
270 // if ((ret = mmap( start, size_low, prot,
271 // flags, fd, offset_low )) != (LPVOID)-1)
272 if ((ret = mmap( start, size_low, prot,
273 MAP_PRIVATE | MAP_FIXED, fd, offset_low )) != (LPVOID)-1)
275 // printf("address %08x\n", *(int*)ret);
276 // printf("%x\n", ret);
277 return ret;
280 // printf("mmap %d\n", errno);
282 /* mmap() failed; if this is because the file offset is not */
283 /* page-aligned (EINVAL), or because the underlying filesystem */
284 /* does not support mmap() (ENOEXEC), we do it by hand. */
286 if (unix_handle == -1) return ret;
287 if ((errno != ENOEXEC) && (errno != EINVAL)) return ret;
288 if (prot & PROT_WRITE)
290 /* We cannot fake shared write mappings */
291 #ifdef MAP_SHARED
292 if (flags & MAP_SHARED) return ret;
293 #endif
294 #ifdef MAP_PRIVATE
295 if (!(flags & MAP_PRIVATE)) return ret;
296 #endif
298 /* printf( "FILE_mmap: mmap failed (%d), faking it\n", errno );*/
299 /* Reserve the memory with an anonymous mmap */
300 ret = FILE_dommap( -1, start, size_high, size_low, 0, 0,
301 PROT_READ | PROT_WRITE, flags );
302 if (ret == (LPVOID)-1)
303 // {
304 // perror(
305 return ret;
306 /* Now read in the file */
307 if ((pos = lseek( fd, offset_low, SEEK_SET )) == -1)
309 FILE_munmap( ret, size_high, size_low );
310 // printf("lseek\n");
311 return (LPVOID)-1;
313 read( fd, ret, size_low );
314 lseek( fd, pos, SEEK_SET ); /* Restore the file pointer */
315 mprotect( ret, size_low, prot ); /* Set the right protection */
316 // printf("address %08x\n", *(int*)ret);
317 return ret;
321 /***********************************************************************
322 * FILE_munmap
324 int FILE_munmap( LPVOID start, DWORD size_high, DWORD size_low )
326 if (size_high)
327 printf("offsets larger than 4Gb not supported\n");
328 return munmap( start, size_low );
330 static int mapping_size=0;
332 struct file_mapping_s;
333 typedef struct file_mapping_s
335 int mapping_size;
336 char* name;
337 LPVOID handle;
338 struct file_mapping_s* next;
339 struct file_mapping_s* prev;
340 }file_mapping;
341 static file_mapping* fm=0;
345 #define PAGE_NOACCESS 0x01
346 #define PAGE_READONLY 0x02
347 #define PAGE_READWRITE 0x04
348 #define PAGE_WRITECOPY 0x08
349 #define PAGE_EXECUTE 0x10
350 #define PAGE_EXECUTE_READ 0x20
351 #define PAGE_EXECUTE_READWRITE 0x40
352 #define PAGE_EXECUTE_WRITECOPY 0x80
353 #define PAGE_GUARD 0x100
354 #define PAGE_NOCACHE 0x200
356 HANDLE WINAPI CreateFileMappingA(HANDLE handle, LPSECURITY_ATTRIBUTES lpAttr,
357 DWORD flProtect,
358 DWORD dwMaxHigh, DWORD dwMaxLow,
359 LPCSTR name)
361 int hFile = (int)handle;
362 unsigned int len;
363 LPVOID answer;
364 int anon=0;
365 int mmap_access=0;
366 if(hFile<0)
368 anon=1;
369 hFile=open("/dev/zero", O_RDWR);
370 if(hFile<0){
371 perror( "Cannot open /dev/zero for READ+WRITE. Check permissions! error: " );
372 return 0;
375 if(!anon)
377 len=lseek(hFile, 0, SEEK_END);
378 lseek(hFile, 0, SEEK_SET);
380 else len=dwMaxLow;
382 if(flProtect & PAGE_READONLY)
383 mmap_access |=PROT_READ;
384 else
385 mmap_access |=PROT_READ|PROT_WRITE;
387 answer=mmap(NULL, len, mmap_access, MAP_PRIVATE, hFile, 0);
388 if(anon)
389 close(hFile);
390 if(answer!=(LPVOID)-1)
392 if(fm==0)
394 fm = (file_mapping*) malloc(sizeof(file_mapping));
395 fm->prev=NULL;
397 else
399 fm->next = (file_mapping*) malloc(sizeof(file_mapping));
400 fm->next->prev=fm;
401 fm=fm->next;
403 fm->next=NULL;
404 fm->handle=answer;
405 if(name)
407 fm->name = (char*) malloc(strlen(name)+1);
408 strcpy(fm->name, name);
410 else
411 fm->name=NULL;
412 fm->mapping_size=len;
414 if(anon)
415 close(hFile);
416 return (HANDLE)answer;
418 return (HANDLE)0;
420 WIN_BOOL WINAPI UnmapViewOfFile(LPVOID handle)
422 file_mapping* p;
423 int result;
424 if(fm==0)
425 return 0;
426 for(p=fm; p; p=p->next)
428 if(p->handle==handle)
430 result=munmap((void*)handle, p->mapping_size);
431 if(p->next)p->next->prev=p->prev;
432 if(p->prev)p->prev->next=p->next;
433 if(p->name)
434 free(p->name);
435 if(p==fm)
436 fm=p->prev;
437 free(p);
438 return result;
441 return 0;
443 //static int va_size=0;
444 struct virt_alloc_s;
445 typedef struct virt_alloc_s
447 int mapping_size;
448 char* address;
449 struct virt_alloc_s* next;
450 struct virt_alloc_s* prev;
451 int state;
452 }virt_alloc;
453 static virt_alloc* vm=0;
454 #define MEM_COMMIT 0x00001000
455 #define MEM_RESERVE 0x00002000
457 LPVOID WINAPI VirtualAlloc(LPVOID address, DWORD size, DWORD type, DWORD protection)
459 void* answer;
460 int fd;
461 long pgsz;
463 //printf("VirtualAlloc(0x%08X, %u, 0x%08X, 0x%08X)\n", (unsigned)address, size, type, protection);
465 if ((type&(MEM_RESERVE|MEM_COMMIT)) == 0) return NULL;
467 fd=open("/dev/zero", O_RDWR);
468 if(fd<0){
469 perror( "Cannot open /dev/zero for READ+WRITE. Check permissions! error: " );
470 return NULL;
473 if (type&MEM_RESERVE && (unsigned)address&0xffff) {
474 size += (unsigned)address&0xffff;
475 (unsigned)address &= ~0xffff;
477 pgsz = sysconf(_SC_PAGESIZE);
478 if (type&MEM_COMMIT && (unsigned)address%pgsz) {
479 size += (unsigned)address%pgsz;
480 address -= (unsigned)address%pgsz;
483 if (type&MEM_RESERVE && size<0x10000) size = 0x10000;
484 if (size%pgsz) size += pgsz - size%pgsz;
486 if(address!=0)
488 //check whether we can allow to allocate this
489 virt_alloc* str=vm;
490 while(str)
492 if((unsigned)address>=(unsigned)str->address+str->mapping_size)
494 str=str->prev;
495 continue;
497 if((unsigned)address+size<=(unsigned)str->address)
499 str=str->prev;
500 continue;
502 if(str->state==0)
504 #warning FIXME
505 if( ((unsigned)address >= (unsigned)str->address)
506 && ((unsigned)address+size<=(unsigned)str->address+str->mapping_size)
507 && (type & MEM_COMMIT))
509 close(fd);
510 return address; //returning previously reserved memory
512 //printf(" VirtualAlloc(...) does not commit or not entirely within reserved, and\n");
514 /*printf(" VirtualAlloc(...) (0x%08X, %u) overlaps with (0x%08X, %u, state=%d)\n",
515 (unsigned)address, size, (unsigned)str->address, str->mapping_size, str->state);*/
516 close(fd);
517 return NULL;
521 answer=mmap(address, size, PROT_READ | PROT_WRITE | PROT_EXEC,
522 MAP_PRIVATE, fd, 0);
523 // answer=FILE_dommap(-1, address, 0, size, 0, 0,
524 // PROT_READ | PROT_WRITE | PROT_EXEC, MAP_PRIVATE);
525 close(fd);
526 if (answer != (void *)-1 && address && answer != address) {
527 /* It is dangerous to try mmap() with MAP_FIXED since it does not
528 always detect conflicts or non-allocation and chaos ensues after
529 a successful call but an overlapping or non-allocated region. */
530 munmap(answer, size);
531 answer = (void *) -1;
532 errno = EINVAL;
533 //printf(" VirtualAlloc(...) cannot satisfy requested address but address=NULL would work.\n");
535 if(answer==(void*)-1)
537 /*printf(" VirtualAlloc(...) mmap(0x%08X, %u, ...) failed with errno=%d (\"%s\")\n",
538 (unsigned)address, size, errno, strerror(errno));*/
539 return NULL;
541 else
543 virt_alloc *new_vm = (virt_alloc*) malloc(sizeof(virt_alloc));
544 new_vm->mapping_size=size;
545 new_vm->address=(char*)answer;
546 new_vm->prev=vm;
547 if(type == MEM_RESERVE)
548 new_vm->state=0;
549 else
550 new_vm->state=1;
551 if(vm)
552 vm->next=new_vm;
553 vm=new_vm;
554 vm->next=0;
555 //if(va_size!=0)
556 // printf("Multiple VirtualAlloc!\n");
557 //printf(" VirtualAlloc(...) provides (0x%08X, %u)\n", (unsigned)answer, size);
558 return answer;
562 WIN_BOOL WINAPI VirtualFree(LPVOID address, SIZE_T dwSize, DWORD dwFreeType)//not sure
564 virt_alloc* str=vm;
565 int answer;
567 //printf("VirtualFree(0x%08X, %d, 0x%08X)\n", (unsigned)address, dwSize, dwFreeType);
568 while(str)
570 if(address!=str->address)
572 str=str->prev;
573 continue;
575 //printf(" VirtualFree(...) munmap(0x%08X, %d)\n", (unsigned)str->address, str->mapping_size);
576 answer=munmap(str->address, str->mapping_size);
577 if(str->next)str->next->prev=str->prev;
578 if(str->prev)str->prev->next=str->next;
579 if(vm==str)vm=str->prev;
580 free(str);
581 return 0;
583 return -1;
586 INT WINAPI WideCharToMultiByte(UINT codepage, DWORD flags, LPCWSTR src,
587 INT srclen,LPSTR dest, INT destlen, LPCSTR defch, WIN_BOOL* used_defch)
589 int i;
590 if(src==0)
591 return 0;
592 if ((srclen==-1)&&(dest==0)) return 0;
593 if(srclen==-1){srclen=0; while(src[srclen++]);}
594 // for(i=0; i<srclen; i++)
595 // printf("%c", src[i]);
596 // printf("\n");
597 if(dest==0)
599 for(i=0; i<srclen; i++)
601 src++;
602 if(*src==0)
603 return i+1;
605 return srclen+1;
607 if(used_defch)
608 *used_defch=0;
609 for(i=0; i<min(srclen, destlen); i++)
611 *dest=(char)*src;
612 dest++;
613 src++;
614 if(*src==0)
615 return i+1;
617 return min(srclen, destlen);
619 INT WINAPI MultiByteToWideChar(UINT codepage,DWORD flags, LPCSTR src, INT srclen,
620 LPWSTR dest, INT destlen)
622 return 0;
624 HANDLE WINAPI OpenFileMappingA(DWORD access, WIN_BOOL prot, LPCSTR name)
626 file_mapping* p;
627 if(fm==0)
628 return (HANDLE)0;
629 if(name==0)
630 return (HANDLE)0;
631 for(p=fm; p; p=p->prev)
633 if(p->name==0)
634 continue;
635 if(strcmp(p->name, name)==0)
636 return (HANDLE)p->handle;
638 return 0;