mbuf.c 5.0 KB
Newer Older
B
bellard 已提交
1 2 3 4 5 6 7 8 9 10 11 12
/*
 * Copyright (c) 1995 Danny Gasparovski
 *
 * Please read the file COPYRIGHT for the
 * terms and conditions of the copyright.
 */

/*
 * mbuf's in SLiRP are much simpler than the real mbufs in
 * FreeBSD.  They are fixed size, determined by the MTU,
 * so that one whole packet can fit.  Mbuf's cannot be
 * chained together.  If there's more data than the mbuf
13
 * could hold, an external g_malloced buffer is pointed to
B
bellard 已提交
14 15 16 17
 * by m_ext (and the data pointers) and M_EXT is set in
 * the flags
 */

P
Peter Maydell 已提交
18
#include "qemu/osdep.h"
19
#include "slirp.h"
B
bellard 已提交
20

21 22 23 24 25
#define MBUF_THRESH 30

/*
 * Find a nice value for msize
 */
26 27
#define SLIRP_MSIZE\
    (offsetof(struct mbuf, m_dat) + IF_MAXLINKHDR + TCPIPHDR_DELTA + IF_MTU)
B
bellard 已提交
28 29

void
30
m_init(Slirp *slirp)
B
bellard 已提交
31
{
32 33
    slirp->m_freelist.qh_link = slirp->m_freelist.qh_rlink = &slirp->m_freelist;
    slirp->m_usedlist.qh_link = slirp->m_usedlist.qh_rlink = &slirp->m_usedlist;
B
bellard 已提交
34 35
}

36 37 38 39
void m_cleanup(Slirp *slirp)
{
    struct mbuf *m, *next;

40 41
    m = (struct mbuf *) slirp->m_usedlist.qh_link;
    while ((struct quehead *) m != &slirp->m_usedlist) {
42 43
        next = m->m_next;
        if (m->m_flags & M_EXT) {
44
            g_free(m->m_ext);
45
        }
46
        g_free(m);
47 48
        m = next;
    }
49 50
    m = (struct mbuf *) slirp->m_freelist.qh_link;
    while ((struct quehead *) m != &slirp->m_freelist) {
51
        next = m->m_next;
52
        g_free(m);
53 54 55 56
        m = next;
    }
}

B
bellard 已提交
57 58
/*
 * Get an mbuf from the free list, if there are none
59
 * allocate one
60
 *
B
bellard 已提交
61 62
 * Because fragmentation can occur if we alloc new mbufs and
 * free old mbufs, we mark all mbufs above mbuf_thresh as M_DOFREE,
63
 * which tells m_free to actually g_free() it
B
bellard 已提交
64 65
 */
struct mbuf *
66
m_get(Slirp *slirp)
B
bellard 已提交
67 68 69
{
	register struct mbuf *m;
	int flags = 0;
70

B
bellard 已提交
71
	DEBUG_CALL("m_get");
72

73
	if (slirp->m_freelist.qh_link == &slirp->m_freelist) {
74
                m = g_malloc(SLIRP_MSIZE);
75 76
		slirp->mbuf_alloced++;
		if (slirp->mbuf_alloced > MBUF_THRESH)
B
bellard 已提交
77
			flags = M_DOFREE;
78
		m->slirp = slirp;
B
bellard 已提交
79
	} else {
80
		m = (struct mbuf *) slirp->m_freelist.qh_link;
B
bellard 已提交
81 82
		remque(m);
	}
83

B
bellard 已提交
84
	/* Insert it in the used list */
85
	insque(m,&slirp->m_usedlist);
B
bellard 已提交
86
	m->m_flags = (flags | M_USEDLIST);
87

B
bellard 已提交
88
	/* Initialise it */
A
Anthony Liguori 已提交
89
	m->m_size = SLIRP_MSIZE - offsetof(struct mbuf, m_dat);
B
bellard 已提交
90 91
	m->m_data = m->m_dat;
	m->m_len = 0;
92 93
        m->m_nextpkt = NULL;
        m->m_prevpkt = NULL;
94
        m->resolution_requested = false;
F
Fabien Chouteau 已提交
95
        m->expiration_date = (uint64_t)-1;
96
	DEBUG_ARG("m = %p", m);
B
bellard 已提交
97 98 99 100
	return m;
}

void
101
m_free(struct mbuf *m)
B
bellard 已提交
102
{
103

B
bellard 已提交
104
  DEBUG_CALL("m_free");
105
  DEBUG_ARG("m = %p", m);
106

B
bellard 已提交
107 108 109 110
  if(m) {
	/* Remove from m_usedlist */
	if (m->m_flags & M_USEDLIST)
	   remque(m);
111

B
bellard 已提交
112
	/* If it's M_EXT, free() it */
113 114 115
        if (m->m_flags & M_EXT) {
                g_free(m->m_ext);
        }
B
bellard 已提交
116 117 118 119
	/*
	 * Either free() it or put it on the free list
	 */
	if (m->m_flags & M_DOFREE) {
120
		m->slirp->mbuf_alloced--;
121
                g_free(m);
B
bellard 已提交
122
	} else if ((m->m_flags & M_FREELIST) == 0) {
123
		insque(m,&m->slirp->m_freelist);
B
bellard 已提交
124 125 126 127 128 129 130
		m->m_flags = M_FREELIST; /* Clobber other flags */
	}
  } /* if(m) */
}

/*
 * Copy data from one mbuf to the end of
131
 * the other.. if result is too big for one mbuf, allocate
B
bellard 已提交
132 133 134
 * an M_EXT data segment
 */
void
135
m_cat(struct mbuf *m, struct mbuf *n)
B
bellard 已提交
136 137 138 139 140
{
	/*
	 * If there's no room, realloc
	 */
	if (M_FREEROOM(m) < n->m_len)
141
		m_inc(m, m->m_len + n->m_len);
142

B
bellard 已提交
143 144 145 146 147 148 149
	memcpy(m->m_data+m->m_len, n->m_data, n->m_len);
	m->m_len += n->m_len;

	m_free(n);
}


150
/* make m 'size' bytes large from m_data */
B
bellard 已提交
151
void
152
m_inc(struct mbuf *m, int size)
B
bellard 已提交
153
{
154
    int gapsize;
B
bellard 已提交
155

P
Prasad J Pandit 已提交
156
    /* some compilers throw up on gotos.  This one we can fake. */
157
    if (M_ROOM(m) > size) {
P
Prasad J Pandit 已提交
158 159
        return;
    }
B
bellard 已提交
160

P
Prasad J Pandit 已提交
161
    if (m->m_flags & M_EXT) {
162 163
        gapsize = m->m_data - m->m_ext;
        m->m_ext = g_realloc(m->m_ext, size + gapsize);
P
Prasad J Pandit 已提交
164
    } else {
165 166
        gapsize = m->m_data - m->m_dat;
        m->m_ext = g_malloc(size + gapsize);
P
Prasad J Pandit 已提交
167 168 169
        memcpy(m->m_ext, m->m_dat, m->m_size);
        m->m_flags |= M_EXT;
    }
170

171 172
    m->m_data = m->m_ext + gapsize;
    m->m_size = size + gapsize;
B
bellard 已提交
173 174 175 176 177
}



void
178
m_adj(struct mbuf *m, int len)
B
bellard 已提交
179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
{
	if (m == NULL)
		return;
	if (len >= 0) {
		/* Trim from head */
		m->m_data += len;
		m->m_len -= len;
	} else {
		/* Trim from tail */
		len = -len;
		m->m_len -= len;
	}
}


/*
 * Copy len bytes from m, starting off bytes into n
 */
int
198
m_copy(struct mbuf *n, struct mbuf *m, int off, int len)
B
bellard 已提交
199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
{
	if (len > M_FREEROOM(n))
		return -1;

	memcpy((n->m_data + n->m_len), (m->m_data + off), len);
	n->m_len += len;
	return 0;
}


/*
 * Given a pointer into an mbuf, return the mbuf
 * XXX This is a kludge, I should eliminate the need for it
 * Fortunately, it's not used often
 */
struct mbuf *
215
dtom(Slirp *slirp, void *dat)
B
bellard 已提交
216 217
{
	struct mbuf *m;
218

B
bellard 已提交
219
	DEBUG_CALL("dtom");
220
	DEBUG_ARG("dat = %p", dat);
B
bellard 已提交
221 222

	/* bug corrected for M_EXT buffers */
223 224
	for (m = (struct mbuf *) slirp->m_usedlist.qh_link;
	     (struct quehead *) m != &slirp->m_usedlist;
225
	     m = m->m_next) {
B
bellard 已提交
226 227 228 229 230 231 232 233
	  if (m->m_flags & M_EXT) {
	    if( (char *)dat>=m->m_ext && (char *)dat<(m->m_ext + m->m_size) )
	      return m;
	  } else {
	    if( (char *)dat >= m->m_dat && (char *)dat<(m->m_dat + m->m_size) )
	      return m;
	  }
	}
234

B
bellard 已提交
235
	DEBUG_ERROR((dfd, "dtom failed"));
236

B
bellard 已提交
237 238
	return (struct mbuf *)0;
}