mbuf.c 4.9 KB
Newer Older
B
bellard 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
 * 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
 * could hold, an external malloced buffer is pointed to
 * by m_ext (and the data pointers) and M_EXT is set in
 * the flags
 */

#include <slirp.h>

20 21 22 23 24 25
#define MBUF_THRESH 30

/*
 * Find a nice value for msize
 * XXX if_maxlinkhdr already in mtu
 */
B
Bruce Rogers 已提交
26
#define SLIRP_MSIZE (IF_MTU + IF_MAXLINKHDR + offsetof(struct mbuf, m_dat) + 6)
B
bellard 已提交
27 28

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

35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
void m_cleanup(Slirp *slirp)
{
    struct mbuf *m, *next;

    m = slirp->m_usedlist.m_next;
    while (m != &slirp->m_usedlist) {
        next = m->m_next;
        if (m->m_flags & M_EXT) {
            free(m->m_ext);
        }
        free(m);
        m = next;
    }
    m = slirp->m_freelist.m_next;
    while (m != &slirp->m_freelist) {
        next = m->m_next;
        free(m);
        m = next;
    }
}

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

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

72
	if (slirp->m_freelist.m_next == &slirp->m_freelist) {
73
		m = (struct mbuf *)malloc(SLIRP_MSIZE);
B
bellard 已提交
74
		if (m == NULL) goto end_error;
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 = slirp->m_freelist.m_next;
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;
B
bellard 已提交
96
end_error:
97
	DEBUG_ARG("m = %p", m);
B
bellard 已提交
98 99 100 101
	return m;
}

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

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

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

B
bellard 已提交
113 114 115 116 117 118 119 120
	/* If it's M_EXT, free() it */
	if (m->m_flags & M_EXT)
	   free(m->m_ext);

	/*
	 * Either free() it or put it on the free list
	 */
	if (m->m_flags & M_DOFREE) {
121
		m->slirp->mbuf_alloced--;
M
Mark McLoughlin 已提交
122
		free(m);
B
bellard 已提交
123
	} else if ((m->m_flags & M_FREELIST) == 0) {
124
		insque(m,&m->slirp->m_freelist);
B
bellard 已提交
125 126 127 128 129 130 131 132 133 134 135
		m->m_flags = M_FREELIST; /* Clobber other flags */
	}
  } /* if(m) */
}

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

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

	m_free(n);
}


/* make m size bytes large */
void
153
m_inc(struct mbuf *m, int size)
B
bellard 已提交
154
{
B
bellard 已提交
155 156
	int datasize;

B
bellard 已提交
157 158 159 160
	/* some compiles throw up on gotos.  This one we can fake. */
        if(m->m_size>size) return;

        if (m->m_flags & M_EXT) {
B
bellard 已提交
161
	  datasize = m->m_data - m->m_ext;
B
bellard 已提交
162
	  m->m_ext = (char *)realloc(m->m_ext,size);
B
bellard 已提交
163
	  m->m_data = m->m_ext + datasize;
B
bellard 已提交
164 165 166 167 168
        } else {
	  char *dat;
	  datasize = m->m_data - m->m_dat;
	  dat = (char *)malloc(size);
	  memcpy(dat, m->m_dat, m->m_size);
169

B
bellard 已提交
170 171 172 173
	  m->m_ext = dat;
	  m->m_data = m->m_ext + datasize;
	  m->m_flags |= M_EXT;
        }
174

B
bellard 已提交
175 176 177 178 179 180 181
        m->m_size = size;

}



void
182
m_adj(struct mbuf *m, int len)
B
bellard 已提交
183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201
{
	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
202
m_copy(struct mbuf *n, struct mbuf *m, int off, int len)
B
bellard 已提交
203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218
{
	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 *
219
dtom(Slirp *slirp, void *dat)
B
bellard 已提交
220 221
{
	struct mbuf *m;
222

B
bellard 已提交
223
	DEBUG_CALL("dtom");
224
	DEBUG_ARG("dat = %p", dat);
B
bellard 已提交
225 226

	/* bug corrected for M_EXT buffers */
227 228
	for (m = slirp->m_usedlist.m_next; m != &slirp->m_usedlist;
	     m = m->m_next) {
B
bellard 已提交
229 230 231 232 233 234 235 236
	  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;
	  }
	}
237

B
bellard 已提交
238
	DEBUG_ERROR((dfd, "dtom failed"));
239

B
bellard 已提交
240 241
	return (struct mbuf *)0;
}