2 * Useful functions for winegcc/winewrap
4 * Copyright 2000 Francois Gouget
5 * Copyright 2002 Dimitrie O. Paun
6 * Copyright 2003 Richard Cohen
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23 #include "wine/port.h"
34 # define min(x,y) (((x) < (y)) ? (x) : (y))
39 void error(const char* s
, ...)
44 fprintf(stderr
, "winegcc: ");
45 vfprintf(stderr
, s
, ap
);
50 void* xmalloc(size_t size
)
54 if ((p
= malloc (size
)) == NULL
)
55 error("Could not malloc %d bytes\n", size
);
60 void *xrealloc(void* p
, size_t size
)
62 void* p2
= realloc (p
, size
);
64 error("Could not realloc %d bytes\n", size
);
69 int strendswith(const char* str
, const char* end
)
74 return l
>= m
&& strcmp(str
+ l
- m
, end
) == 0;
77 char* strmake(const char* fmt
, ...)
85 char *p
= xmalloc (size
);
87 n
= vsnprintf (p
, size
, fmt
, ap
);
89 if (n
== -1) size
*= 2;
90 else if ((size_t)n
>= size
) size
= n
+ 1;
96 strarray
* strarray_alloc(void)
98 strarray
* arr
= xmalloc(sizeof(*arr
));
99 arr
->maximum
= arr
->size
= 0;
104 void strarray_free(strarray
* arr
)
110 void strarray_add(strarray
* arr
, const char* str
)
112 if (arr
->size
== arr
->maximum
)
115 arr
->base
= xrealloc(arr
->base
, sizeof(*(arr
->base
)) * arr
->maximum
);
117 arr
->base
[arr
->size
++] = str
;
120 void strarray_del(strarray
* arr
, unsigned int i
)
122 if (i
>= arr
->size
) error("Invalid index i=%d\n", i
);
123 memmove(&arr
->base
[i
], &arr
->base
[i
+ 1], (arr
->size
- i
- 1) * sizeof(arr
->base
[0]));
127 void strarray_addall(strarray
* arr
, const strarray
* from
)
131 for (i
= 0; i
< from
->size
; i
++)
132 strarray_add(arr
, from
->base
[i
]);
135 strarray
* strarray_dup(const strarray
* arr
)
137 strarray
* dup
= strarray_alloc();
140 for (i
= 0; i
< arr
->size
; i
++)
141 strarray_add(dup
, arr
->base
[i
]);
146 strarray
* strarray_fromstring(const char* str
, const char* delim
)
148 strarray
* arr
= strarray_alloc();
149 char* buf
= strdup(str
);
152 for(tok
= strtok(buf
, delim
); tok
; tok
= strtok(0, delim
))
153 strarray_add(arr
, strdup(tok
));
159 char* strarray_tostring(const strarray
* arr
, const char* sep
)
164 str
= strmake("%s", arr
->base
[0]);
165 for (i
= 1; i
< arr
->size
; i
++)
167 newstr
= strmake("%s%s%s", str
, sep
, arr
->base
[i
]);
175 char* get_basename(const char* file
)
180 if ((name
= strrchr(file
, '/'))) name
++;
183 base_name
= strdup(name
);
184 if ((p
= strrchr(base_name
, '.'))) *p
= 0;
189 void create_file(const char* name
, int mode
, const char* fmt
, ...)
194 if (verbose
) printf("Creating file %s\n", name
);
196 if ( !(file
= fopen(name
, "w")) )
197 error("Unable to open %s for writing\n", name
);
198 vfprintf(file
, fmt
, ap
);
204 file_type
get_file_type(const char* filename
)
206 /* see tools/winebuild/res32.c: check_header for details */
207 static const char res_sig
[] = { 0,0,0,0, 32,0,0,0, 0xff,0xff, 0,0, 0xff,0xff, 0,0, 0,0,0,0, 0,0, 0,0, 0,0,0,0, 0,0,0,0 };
208 char buf
[sizeof(res_sig
)];
211 fd
= open( filename
, O_RDONLY
);
212 if (fd
== -1) return file_na
;
213 cnt
= read(fd
, buf
, sizeof(buf
));
215 if (cnt
== -1) return file_na
;
217 if (cnt
== sizeof(res_sig
) && !memcmp(buf
, res_sig
, sizeof(res_sig
))) return file_res
;
218 if (strendswith(filename
, ".o")) return file_obj
;
219 if (strendswith(filename
, ".a")) return file_arh
;
220 if (strendswith(filename
, ".res")) return file_res
;
221 if (strendswith(filename
, ".so")) return file_so
;
222 if (strendswith(filename
, ".dylib")) return file_so
;
223 if (strendswith(filename
, ".def")) return file_def
;
224 if (strendswith(filename
, ".spec")) return file_spec
;
225 if (strendswith(filename
, ".rc")) return file_rc
;
230 static char* try_lib_path(const char* dir
, const char* pre
,
231 const char* library
, const char* ext
,
232 file_type expected_type
)
237 /* first try a subdir named from the library we are looking for */
238 fullname
= strmake("%s/%s/%s%s%s", dir
, library
, pre
, library
, ext
);
239 if (verbose
> 1) fprintf(stderr
, "Try %s...", fullname
);
240 type
= get_file_type(fullname
);
241 if (verbose
> 1) fprintf(stderr
, type
== expected_type
? "FOUND!\n" : "no\n");
242 if (type
== expected_type
) return fullname
;
245 fullname
= strmake("%s/%s%s%s", dir
, pre
, library
, ext
);
246 if (verbose
> 1) fprintf(stderr
, "Try %s...", fullname
);
247 type
= get_file_type(fullname
);
248 if (verbose
> 1) fprintf(stderr
, type
== expected_type
? "FOUND!\n" : "no\n");
249 if (type
== expected_type
) return fullname
;
254 static file_type
guess_lib_type(const char* dir
, const char* library
, char** file
)
256 /* Unix shared object */
257 if ((*file
= try_lib_path(dir
, "lib", library
, ".so", file_so
)))
260 /* Mach-O (Darwin/Mac OS X) Dynamic Library behaves mostly like .so */
261 if ((*file
= try_lib_path(dir
, "lib", library
, ".dylib", file_so
)))
265 if ((*file
= try_lib_path(dir
, "lib", library
, ".def", file_def
)))
267 if ((*file
= try_lib_path(dir
, "", library
, ".def", file_def
)))
270 /* Unix static archives */
271 if ((*file
= try_lib_path(dir
, "lib", library
, ".a", file_arh
)))
277 file_type
get_lib_type(strarray
* path
, const char* library
, char** file
)
281 for (i
= 0; i
< path
->size
; i
++)
283 file_type type
= guess_lib_type(path
->base
[i
], library
, file
);
284 if (type
!= file_na
) return type
;
289 void spawn(const strarray
* prefix
, const strarray
* args
, int ignore_errors
)
293 strarray
* arr
= strarray_dup(args
);
297 strarray_add(arr
, NULL
);
302 for (i
= 0; i
< prefix
->size
; i
++)
307 if (!(p
= strrchr(argv
[0], '/'))) p
= argv
[0];
309 prog
= strmake("%s/%s", prefix
->base
[i
], p
);
310 if (stat(prog
, &st
) == 0 && S_ISREG(st
.st_mode
) && (st
.st_mode
& 0111))
320 for(i
= 0; argv
[i
]; i
++) printf("%s ", argv
[i
]);
324 if ((status
= spawnvp( _P_WAIT
, argv
[0], argv
)) && !ignore_errors
)
326 if (status
> 0) error("%s failed\n", argv
[0]);
327 else perror("winegcc");