1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
/* Support for compressed modules. Willy Tarreau <willy@meta-x.org>
* did the support for modutils, Andrey Borzenkov <arvidjaar@mail.ru>
* ported it to module-init-tools, and I said it was too ugly to live
* and rewrote it 8).
*
* (C) 2003 Rusty Russell, IBM Corporation.
*/
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/mman.h>
#include "zlibsupport.h"
#ifdef CONFIG_USE_ZLIB
#include <zlib.h>
void *grab_contents(gzFile *gzfd, unsigned long *size)
{
unsigned int max = 16384;
void *buffer = malloc(max);
int ret;
if (!buffer)
return NULL;
*size = 0;
while ((ret = gzread(gzfd, buffer + *size, max - *size)) > 0) {
*size += ret;
if (*size == max) {
void *p;
p = realloc(buffer, max *= 2);
if (!p)
goto out_err;
buffer = p;
}
}
if (ret < 0)
goto out_err;
return buffer;
out_err:
free(buffer);
return NULL;
}
void *grab_fd(int fd, unsigned long *size)
{
gzFile gzfd;
gzfd = gzdopen(fd, "rb");
if (!gzfd)
return NULL;
/* gzclose(gzfd) would close fd, which would drop locks.
Don't blame zlib: POSIX locking semantics are so horribly
broken that they should be ripped out. */
return grab_contents(gzfd, size);
}
/* gzopen handles uncompressed files transparently. */
void *grab_file(const char *filename, unsigned long *size)
{
gzFile gzfd;
void *buffer;
gzfd = gzopen(filename, "rb");
if (!gzfd)
return NULL;
buffer = grab_contents(gzfd, size);
gzclose(gzfd);
return buffer;
}
void release_file(void *data, unsigned long size)
{
free(data);
}
#else /* ... !CONFIG_USE_ZLIB */
void *grab_fd(int fd, unsigned long *size)
{
struct stat st;
void *map;
int ret;
ret = fstat(fd, &st);
if (ret < 0)
return NULL;
*size = st.st_size;
map = mmap(0, *size, PROT_READ|PROT_WRITE, MAP_PRIVATE, fd, 0);
if (map == MAP_FAILED)
map = NULL;
return map;
}
void *grab_file(const char *filename, unsigned long *size)
{
int fd;
void *map;
fd = open(filename, O_RDONLY, 0);
if (fd < 0)
return NULL;
map = grab_fd(fd, size);
close(fd);
return map;
}
void release_file(void *data, unsigned long size)
{
munmap(data, size);
}
#endif
|