xref: /haiku/src/libs/compat/freebsd_network/mbuf.c (revision eea5774f46bba925156498abf9cb1a1165647bf7)
1 /*
2  * Copyright 2007, Hugo Santos. All Rights Reserved.
3  * Copyright 2004, Marcus Overhagen. All Rights Reserved.
4  * Distributed under the terms of the MIT License.
5  */
6 
7 
8 #include "device.h"
9 
10 #include <stdint.h>
11 #include <string.h>
12 #include <slab/Slab.h>
13 
14 #include <net_buffer.h>
15 
16 #include <compat/sys/malloc.h>
17 #include <compat/sys/mbuf.h>
18 #include <compat/sys/kernel.h>
19 
20 
21 static object_cache *sMBufCache;
22 static object_cache *sChunkCache;
23 static object_cache *sJumbo9ChunkCache;
24 static object_cache *sJumboPageSizeCache;
25 
26 
27 int max_linkhdr = 16;
28 int max_protohdr = 40 + 20; /* ip6 + tcp */
29 
30 /* max_linkhdr + max_protohdr, but that's not allowed by gcc. */
31 int max_hdr = 16 + 40 + 20;
32 
33 /* MHLEN - max_hdr */
34 int max_datalen = MHLEN - (16 + 40 + 20);
35 
36 
37 static int
38 m_to_oc_flags(int how)
39 {
40 	if (how & M_NOWAIT)
41 		return CACHE_DONT_WAIT_FOR_MEMORY;
42 
43 	return 0;
44 }
45 
46 
47 int
48 m_init(struct mbuf *m, int how, short type, int flags)
49 {
50 	int error;
51 
52 	if (type == MT_NOINIT)
53 		return 0;
54 
55 	m->m_next = NULL;
56 	m->m_nextpkt = NULL;
57 	m->m_data = m->m_dat;
58 	m->m_len = 0;
59 	m->m_flags = flags;
60 	m->m_type = type;
61 	if (flags & M_PKTHDR)
62 		error = m_pkthdr_init(m, how);
63 	else
64 		error = 0;
65 
66 	return (error);
67 }
68 
69 
70 static void*
71 allocate_ext_buf(int how, int size, int* ext_type)
72 {
73 	object_cache *cache;
74 	int extType;
75 	if (size != MCLBYTES && size != MJUM9BYTES && size != MJUMPAGESIZE)
76 		panic("unsupported size");
77 
78 	if (size == MCLBYTES) {
79 		cache = sChunkCache;
80 		extType = EXT_CLUSTER;
81 	} else if (size == MJUM9BYTES) {
82 		cache = sJumbo9ChunkCache;
83 		extType = EXT_JUMBO9;
84 	} else {
85 		cache = sJumboPageSizeCache;
86 		extType = EXT_JUMBOP;
87 	}
88 
89 	if (ext_type != NULL)
90 		*ext_type = extType;
91 	return object_cache_alloc(cache, m_to_oc_flags(how));
92 }
93 
94 
95 static int
96 construct_ext_sized_mbuf(struct mbuf *memoryBuffer, int how, int size)
97 {
98 	int extType;
99 
100 	memoryBuffer->m_ext.ext_buf = allocate_ext_buf(how, size, &extType);
101 	if (memoryBuffer->m_ext.ext_buf == NULL)
102 		return B_NO_MEMORY;
103 
104 	memoryBuffer->m_data = memoryBuffer->m_ext.ext_buf;
105 	memoryBuffer->m_flags |= M_EXT;
106 	memoryBuffer->m_ext.ext_size = size;
107 	memoryBuffer->m_ext.ext_type = extType;
108 	memoryBuffer->m_ext.ext_flags = EXT_FLAG_EMBREF;
109 	memoryBuffer->m_ext.ext_count = 1;
110 
111 	return 0;
112 }
113 
114 
115 static inline int
116 construct_ext_mbuf(struct mbuf *memoryBuffer, int how)
117 {
118 	return construct_ext_sized_mbuf(memoryBuffer, how, MCLBYTES);
119 }
120 
121 
122 static int
123 construct_pkt_mbuf(int how, struct mbuf *memoryBuffer, short type, int flags)
124 {
125 	if (m_init(memoryBuffer, how, type, flags) < 0)
126 		return -1;
127 	if (construct_ext_mbuf(memoryBuffer, how) < 0)
128 		return -1;
129 	memoryBuffer->m_ext.ext_type = EXT_CLUSTER;
130 	return 0;
131 }
132 
133 
134 struct mbuf *
135 m_getcl(int how, short type, int flags)
136 {
137 	struct mbuf *memoryBuffer =
138 		(struct mbuf *)object_cache_alloc(sMBufCache, m_to_oc_flags(how));
139 	if (memoryBuffer == NULL)
140 		return NULL;
141 
142 	if (construct_pkt_mbuf(how, memoryBuffer, type, flags) < 0) {
143 		object_cache_free(sMBufCache, memoryBuffer, 0);
144 		return NULL;
145 	}
146 
147 	return memoryBuffer;
148 }
149 
150 
151 static struct mbuf *
152 _m_get(int how, short type, int flags)
153 {
154 	struct mbuf *memoryBuffer =
155 		(struct mbuf *)object_cache_alloc(sMBufCache, m_to_oc_flags(how));
156 	if (memoryBuffer == NULL)
157 		return NULL;
158 
159 	m_init(memoryBuffer, how, type, flags);
160 
161 	return memoryBuffer;
162 }
163 
164 
165 struct mbuf *
166 m_get(int how, short type)
167 {
168 	return _m_get(how, type, 0);
169 }
170 
171 
172 struct mbuf *
173 m_get2(int size, int how, short type, int flags)
174 {
175 	if (size <= MHLEN || (size <= MLEN && (flags & M_PKTHDR) == 0)) {
176 		return _m_get(how, type, flags);
177 	} else if (size <= MCLBYTES) {
178 		size = MCLBYTES;
179 	} else if (size <= MJUMPAGESIZE) {
180 		size = MJUMPAGESIZE;
181 	} else if (size <= MJUM9BYTES) {
182 		size = MJUM9BYTES;
183 	} else /* (size > MJUM9BYTES) */ {
184 		return NULL;
185 	}
186 
187 	return m_getjcl(how, type, flags, size);
188 }
189 
190 
191 struct mbuf *
192 m_gethdr(int how, short type)
193 {
194 	return _m_get(how, type, M_PKTHDR);
195 }
196 
197 
198 struct mbuf *
199 m_getjcl(int how, short type, int flags, int size)
200 {
201 	struct mbuf *memoryBuffer =
202 		(struct mbuf *)object_cache_alloc(sMBufCache, m_to_oc_flags(how));
203 	if (memoryBuffer == NULL)
204 		return NULL;
205 	if (m_init(memoryBuffer, how, type, flags) < 0) {
206 		object_cache_free(sMBufCache, memoryBuffer, 0);
207 		return NULL;
208 	}
209 	if (construct_ext_sized_mbuf(memoryBuffer, how, size) < 0) {
210 		object_cache_free(sMBufCache, memoryBuffer, 0);
211 		return NULL;
212 	}
213 	return memoryBuffer;
214 }
215 
216 
217 int
218 m_clget(struct mbuf *memoryBuffer, int how)
219 {
220 	memoryBuffer->m_ext.ext_buf = NULL;
221 	/* called checks for errors by looking for M_EXT */
222 	construct_ext_mbuf(memoryBuffer, how);
223 	return memoryBuffer->m_flags & M_EXT;
224 }
225 
226 
227 void*
228 m_cljget(struct mbuf* memoryBuffer, int how, int size)
229 {
230 	if (memoryBuffer == NULL)
231 		return allocate_ext_buf(how, size, NULL);
232 
233 	memoryBuffer->m_ext.ext_buf = NULL;
234 	construct_ext_sized_mbuf(memoryBuffer, how, size);
235 	return memoryBuffer->m_ext.ext_buf;
236 }
237 
238 
239 static void
240 mb_free_ext(struct mbuf *memoryBuffer)
241 {
242 	volatile u_int *refcnt;
243 	struct mbuf *mref;
244 	int freembuf;
245 
246 	KASSERT(memoryBuffer->m_flags & M_EXT, ("%s: M_EXT not set on %p",
247 		__func__, memoryBuffer));
248 
249 	/* See if this is the mbuf that holds the embedded refcount. */
250 	if (memoryBuffer->m_ext.ext_flags & EXT_FLAG_EMBREF) {
251 		refcnt = &memoryBuffer->m_ext.ext_count;
252 		mref = memoryBuffer;
253 	} else {
254 		KASSERT(memoryBuffer->m_ext.ext_cnt != NULL,
255 			("%s: no refcounting pointer on %p", __func__, memoryBuffer));
256 		refcnt = memoryBuffer->m_ext.ext_cnt;
257 		mref = __containerof(refcnt, struct mbuf, m_ext.ext_count);
258 	}
259 
260 	/*
261 	 * Check if the header is embedded in the cluster.  It is
262 	 * important that we can't touch any of the mbuf fields
263 	 * after we have freed the external storage, since mbuf
264 	 * could have been embedded in it.  For now, the mbufs
265 	 * embedded into the cluster are always of type EXT_EXTREF,
266 	 * and for this type we won't free the mref.
267 	 */
268 	if (memoryBuffer->m_flags & M_NOFREE) {
269 		freembuf = 0;
270 		KASSERT(memoryBuffer->m_ext.ext_type == EXT_EXTREF,
271 			("%s: no-free mbuf %p has wrong type", __func__, memoryBuffer));
272 	} else
273 		freembuf = 1;
274 
275 	/* Free attached storage only if this mbuf is the only reference to it. */
276 	if (*refcnt == 1 || atomic_add((int32*)refcnt, -1) == 1) {
277 		object_cache *cache = NULL;
278 
279 		if (memoryBuffer->m_ext.ext_type == EXT_CLUSTER)
280 			cache = sChunkCache;
281 		else if (memoryBuffer->m_ext.ext_type == EXT_JUMBO9)
282 			cache = sJumbo9ChunkCache;
283 		else if (memoryBuffer->m_ext.ext_type == EXT_JUMBOP)
284 			cache = sJumboPageSizeCache;
285 		else
286 			panic("unknown mbuf ext_type %d", memoryBuffer->m_ext.ext_type);
287 
288 		object_cache_free(cache, memoryBuffer->m_ext.ext_buf, 0);
289 		object_cache_free(sMBufCache, mref, 0);
290 	}
291 
292 	if (freembuf && memoryBuffer != mref)
293 		object_cache_free(sMBufCache, memoryBuffer, 0);
294 }
295 
296 
297 struct mbuf *
298 m_free(struct mbuf* memoryBuffer)
299 {
300 	struct mbuf* next = memoryBuffer->m_next;
301 
302 	if ((memoryBuffer->m_flags & (M_PKTHDR|M_NOFREE)) == (M_PKTHDR|M_NOFREE))
303 		m_tag_delete_chain(memoryBuffer, NULL);
304 	if (memoryBuffer->m_flags & M_EXT)
305 		mb_free_ext(memoryBuffer);
306 	else if ((memoryBuffer->m_flags & M_NOFREE) == 0)
307 		object_cache_free(sMBufCache, memoryBuffer, 0);
308 
309 	return next;
310 }
311 
312 
313 status_t
314 init_mbufs()
315 {
316 	status_t status = get_module(NET_BUFFER_MODULE_NAME, (module_info **)&gBufferModule);
317 	if (status != B_OK)
318 		goto clean;
319 
320 	sMBufCache = create_object_cache("mbufs", MSIZE, 8, NULL, NULL, NULL);
321 	if (sMBufCache == NULL)
322 		goto clean;
323 	sChunkCache = create_object_cache("mbuf chunks", MCLBYTES, 0, NULL, NULL,
324 		NULL);
325 	if (sChunkCache == NULL)
326 		goto clean;
327 	sJumbo9ChunkCache = create_object_cache("mbuf jumbo9 chunks", MJUM9BYTES, 0,
328 		NULL, NULL, NULL);
329 	if (sJumbo9ChunkCache == NULL)
330 		goto clean;
331 	sJumboPageSizeCache = create_object_cache("mbuf jumbo page size chunks",
332 		MJUMPAGESIZE, 0, NULL, NULL, NULL);
333 	if (sJumboPageSizeCache == NULL)
334 		goto clean;
335 	return B_OK;
336 
337 clean:
338 	if (sJumbo9ChunkCache != NULL)
339 		delete_object_cache(sJumbo9ChunkCache);
340 	if (sChunkCache != NULL)
341 		delete_object_cache(sChunkCache);
342 	if (sMBufCache != NULL)
343 		delete_object_cache(sMBufCache);
344 	put_module(NET_BUFFER_MODULE_NAME);
345 	return B_NO_MEMORY;
346 }
347 
348 
349 void
350 uninit_mbufs()
351 {
352 	delete_object_cache(sMBufCache);
353 	delete_object_cache(sChunkCache);
354 	delete_object_cache(sJumbo9ChunkCache);
355 	delete_object_cache(sJumboPageSizeCache);
356 }
357