CSAPP Labs 小结
2023-09-04 12:00:23 # Book

data

bitXor

本质上是使用 非门 ~ 和 与门 & 实现 异或门 ^

$$
\begin{align*}
a \ \oplus \ b &= (a \overline b) + (\overline a b) \
&= \overline{ \overline{ a \overline b} \cdot \overline { \overline a b}}

\end{align*}
$$

1
2
3
int bitXor(int x, int y) {
return ~(~(~x & y) & ~(~y & x));
}

tmin

使用二进制补码, 最高位为 1 时得到 tmin

1
2
3
int tmin(void) {
return 1 << 31;
}

malloc

隐式空闲链表

根据 headfoot 中记录的长度可以找到左右块, 从而组成双向链表

使用策略: 首次适配, 立即合并

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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <unistd.h>
#include <string.h>

#include "mm.h"
#include "memlib.h"

team_t team = {
/* Team name */
"ateam",
/* First member's full name */
"lzl",
/* First member's email address */
"emmmm",
/* Second member's full name (leave blank if none) */
"",
/* Second member's email address (leave blank if none) */
""
};

/* single word (4) or double word (8) alignment */
#define ALIGNMENT 8

/* rounds up to the nearest multiple of ALIGNMENT */
#define ALIGN(size) (((size) + (ALIGNMENT-1)) & ~0x7)


#define SIZE_T_SIZE (ALIGN(sizeof(size_t)))

#define MAX(a, b) ((a) > (b) ? a : b)

#define WSIZE 4
#define DSIZE 8
#define CHUNKSIZE (1 << 12)

#define PACK(size, alloc) ((size) | (alloc))

#define GET(p) (*(unsigned int *)(p))
#define PUT(p, val) (*(unsigned int *)(p) = (val))

#define GET_SIZE(p) (GET(p) & ~0x7)
#define GET_ALLOC(p) (GET(p) & 0x1)

#define HDRP(bp) ((char *)(bp) - WSIZE)
#define FTRP(bp) ((char *)(bp) + GET_SIZE(HDRP(bp)) - DSIZE)

#define NEXT_BLKP(bp) ((char *)(bp) + GET_SIZE(((char *)(bp) - WSIZE)))
#define PREV_BLKP(bp) ((char *)(bp) - GET_SIZE(((char *)(bp) - DSIZE)))

static char *heap_listp;

static void *coalesce(void *bp) {
size_t prev_alloc = GET_ALLOC(FTRP(PREV_BLKP(bp)));
size_t next_alloc = GET_ALLOC(HDRP(NEXT_BLKP(bp)));

size_t size = GET_SIZE(HDRP(bp));

if (prev_alloc && next_alloc) return bp;
else if (prev_alloc && !next_alloc) {

size += GET_SIZE(HDRP(NEXT_BLKP(bp)));
PUT(HDRP(bp), PACK(size, 0));
PUT(FTRP(bp), PACK(size, 0));

} else if (!prev_alloc && next_alloc) {

size += GET_SIZE(HDRP(PREV_BLKP(bp)));
PUT(FTRP(bp), PACK(size, 0));
PUT(HDRP(PREV_BLKP(bp)), PACK(size, 0));
bp = PREV_BLKP(bp);

} else {

size += GET_SIZE(HDRP(NEXT_BLKP(bp)));
size += GET_SIZE(HDRP(PREV_BLKP(bp)));

PUT(FTRP(NEXT_BLKP(bp)), PACK(size, 0));
PUT(HDRP(PREV_BLKP(bp)), PACK(size, 0));
bp = PREV_BLKP(bp);
}
return bp;
}

static void *extend_heap(size_t words) {
char *bp;
size_t size;


size = (words % 2) ? (words + 1) * WSIZE : words * WSIZE;

bp = (char *)mem_sbrk(size);

if ((long)bp == -1) return NULL;

PUT(HDRP(bp), PACK(size, 0));
PUT(FTRP(bp), PACK(size, 0));
PUT(HDRP(NEXT_BLKP(bp)), PACK(0, 1));

return coalesce(bp);
}


static void *find_fit(size_t asize) {

void *cur_ptr = heap_listp;
while (GET_SIZE(HDRP(cur_ptr)) != 0) {
if (GET_SIZE(HDRP(cur_ptr)) >= asize &&
!GET_ALLOC(HDRP(cur_ptr)))
return cur_ptr;
cur_ptr = NEXT_BLKP(cur_ptr);
}

return NULL;
}

static void place(void *bp, size_t asize) {

size_t rest_size = GET_SIZE(HDRP(bp)) - asize;
char *cur_bp = (char *)bp;

if (rest_size >= 2 * DSIZE) {

PUT(HDRP(bp), PACK(asize, 1));
PUT(FTRP(bp), PACK(asize, 1));

bp = NEXT_BLKP(bp);

PUT(HDRP(bp), PACK(rest_size, 0));
PUT(FTRP(bp), PACK(rest_size, 0));

} else {

asize = GET_SIZE(HDRP(bp));
PUT(HDRP(cur_bp), PACK(asize, 1));
PUT(FTRP(cur_bp), PACK(asize, 1));
}
}
/*
* mm_init - initialize the malloc package.
*/
int mm_init(void)
{
mem_init();
if ((heap_listp = (char *)mem_sbrk(4 * WSIZE)) == (void *)-1) return -1;

PUT(heap_listp, 0);
PUT(heap_listp + (1 * WSIZE), PACK(DSIZE, 1));
PUT(heap_listp + (2 * WSIZE), PACK(DSIZE, 1));
PUT(heap_listp + (3 * WSIZE), PACK(0, 1));
heap_listp += (2 * WSIZE);

return 0;
}

/*
* mm_malloc - Allocate a block by incrementing the brk pointer.
* Always allocate a block whose size is a multiple of the alignment.
*/
void *mm_malloc(size_t size) {

size_t asize;
size_t extendsize;
char *bp;

if (size == 0) return NULL;

if (size <= DSIZE) asize = 2 * DSIZE;
else asize = DSIZE * ((size + (DSIZE) + (DSIZE - 1)) / DSIZE);

bp = (char *)find_fit(asize);

if (bp != NULL) {
place(bp, asize);
return bp;
}

extendsize = asize;

bp = (char *)extend_heap(extendsize / WSIZE);

if (bp == NULL) return NULL;

place(bp, asize);

return bp;
}

/*
* mm_free - Freeing a block does nothing.
*/
void mm_free(void *ptr)
{
if (ptr == NULL) return;

size_t size = GET_SIZE(HDRP(ptr));

PUT(HDRP(ptr), PACK(size, 0));
PUT(FTRP(ptr), PACK(size, 0));

coalesce(ptr);
}

/*
* mm_realloc - Implemented simply in terms of mm_malloc and mm_free
*/
void *mm_realloc(void *ptr, size_t size)
{
if (size == 0) {
mm_free(ptr);
return NULL;
}

if (ptr == NULL) {
return mm_malloc(size);
}

void *newptr = mm_malloc(size);
if (!newptr) return NULL;

size_t oldsize = GET_SIZE(HDRP(ptr));

if (size < oldsize) oldsize = size;

memcpy(newptr, ptr, oldsize);
mm_free(ptr);
return newptr;

}

得分 46 (util) + 36 (thru) = 83/100

显式空闲链表

空闲块中 head 之后的两个位置放指针, 指向上一个 / 下一个空闲块

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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
/*
* mm-naive.c - The fastest, least memory-efficient malloc package.
*
* In this naive approach, a block is allocated by simply incrementing
* the brk pointer. A block is pure payload. There are no headers or
* footers. Blocks are never coalesced or reused. Realloc is
* implemented directly using mm_malloc and mm_free.
*
* NOTE TO STUDENTS: Replace this header comment with your own header
* comment that gives a high level description of your solution.
*/
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <unistd.h>
#include <string.h>

#include "mm.h"
#include "memlib.h"

/*********************************************************
* NOTE TO STUDENTS: Before you do anything else, please
* provide your team information in the following struct.
********************************************************/
team_t team = { /* Team name */
"ateam",
/* First member's full name */
"lzl",
/* First member's email address */
"emmmm",
/* Second member's full name (leave blank if none) */
"",
/* Second member's email address (leave blank if none) */
""
};

/* single word (4) or double word (8) alignment */
#define ALIGNMENT 8

/* rounds up to the nearest multiple of ALIGNMENT */
#define ALIGN(size) (((size) + (ALIGNMENT-1)) & ~0x7)


#define SIZE_T_SIZE (ALIGN(sizeof(size_t)))

#define MAX(a, b) ((a) > (b) ? a : b)

#define WSIZE 4
#define DSIZE 8
#define CHUNKSIZE (1 << 6)

#define PACK(size, alloc) ((size) | (alloc))

#define GET(p) (*(unsigned int *)(p))
#define PUT(p, val) (*(unsigned int *)(p) = (val))

#define GET_SIZE(p) (GET(p) & ~0x7)
#define GET_ALLOC(p) (GET(p) & 0x1)

#define HDRP(bp) ((char *)(bp) - WSIZE)
#define FTRP(bp) ((char *)(bp) + GET_SIZE(HDRP(bp)) - DSIZE)

#define NEXT_BLKP(bp) ((char *)(bp) + GET_SIZE(((char *)(bp) - WSIZE)))
#define PREV_BLKP(bp) ((char *)(bp) - GET_SIZE(((char *)(bp) - DSIZE)))

#define GET_PREV(p) (*(unsigned int *)(p))
#define SET_PREV(p, prev) (*(unsigned int *)(p) = (prev))
#define GET_NEXT(p) (*((unsigned int *)(p)+1))
#define SET_NEXT(p, val) (*((unsigned int *)(p)+1) = (val))

static char *heap_listp;
static char *free_listp;

void remove_from_free_list(void *bp) {
if (bp == NULL || GET_ALLOC(bp)) return;

void *prev = GET_PREV(bp);
void *next = GET_NEXT(bp);

SET_PREV(bp, 0), SET_NEXT(bp, 0);

if (prev == NULL && next == NULL) {
free_listp = NULL;
} else if (prev == NULL) {
SET_PREV(next, 0);
free_listp = next;
} else if (next == NULL) {
SET_NEXT(prev, 0);
} else {
SET_NEXT(prev, next);
SET_PREV(next, prev);
}

}

void insert_to_free_list(void *bp) {
if (bp == NULL) return;

if (free_listp == NULL)
return free_listp = bp, (void)0;

SET_NEXT(bp, free_listp);
SET_PREV(free_listp, bp);

free_listp = bp;
}


static void *coalesce(void *bp) {
size_t prev_alloc = GET_ALLOC(FTRP(PREV_BLKP(bp)));
size_t next_alloc = GET_ALLOC(HDRP(NEXT_BLKP(bp)));

size_t size = GET_SIZE(HDRP(bp));

void *prev_bp = PREV_BLKP(bp);
void *next_bp = NEXT_BLKP(bp);

if (prev_alloc && next_alloc) {

}
else if (prev_alloc && !next_alloc) {

remove_from_free_list(next_bp);

size += GET_SIZE(HDRP(next_bp));
PUT(HDRP(bp), PACK(size, 0));
PUT(FTRP(bp), PACK(size, 0));

} else if (!prev_alloc && next_alloc) {

remove_from_free_list(prev_bp);

size += GET_SIZE(HDRP(prev_bp));

PUT(FTRP(bp), PACK(size, 0));
PUT(HDRP(prev_bp), PACK(size, 0));

bp = PREV_BLKP(bp);

} else {

remove_from_free_list(prev_bp);
remove_from_free_list(next_bp);

size += GET_SIZE(HDRP(next_bp));
size += GET_SIZE(HDRP(prev_bp));

PUT(FTRP(next_bp), PACK(size, 0));
PUT(HDRP(prev_bp), PACK(size, 0));
bp = PREV_BLKP(bp);

}
insert_to_free_list(bp);
return bp;
}

static void *extend_heap(size_t words) {
char *bp;
size_t size;

size = (words % 2) ? (words + 1) * WSIZE : words * WSIZE;

bp = (char *)mem_sbrk(size);

if ((long)bp == -1) return NULL;

PUT(HDRP(bp), PACK(size, 0));
PUT(FTRP(bp), PACK(size, 0));
PUT(HDRP(NEXT_BLKP(bp)), PACK(0, 1));

SET_PREV(bp, 0), SET_NEXT(bp, 0);

return coalesce(bp);
}

static void* find_fit(size_t asize) {

for (void* bp = free_listp; bp != 0; bp = GET_NEXT(bp))
if (GET_SIZE(HDRP(bp)) >= asize)
return bp;

return NULL;
}

static void place(void *bp, size_t asize) {

size_t rest_size = GET_SIZE(HDRP(bp)) - asize;
char *cur_bp = (char *)bp;

remove_from_free_list(bp);

if (rest_size >= 2 * DSIZE) {

PUT(HDRP(bp), PACK(asize, 1));
PUT(FTRP(bp), PACK(asize, 1));

bp = NEXT_BLKP(bp);

PUT(HDRP(bp), PACK(rest_size, 0));
PUT(FTRP(bp), PACK(rest_size, 0));

SET_PREV(bp, 0), SET_NEXT(bp, 0);
coalesce(bp);

} else {

asize = GET_SIZE(HDRP(bp));
PUT(HDRP(cur_bp), PACK(asize, 1));
PUT(FTRP(cur_bp), PACK(asize, 1));
}
}
/*
* mm_init - initialize the malloc package.
*/
int mm_init(void)
{
mem_init();
if ((heap_listp = (char *)mem_sbrk(4 * WSIZE)) == (void *)-1) return -1;

PUT(heap_listp, 0);
PUT(heap_listp + (1 * WSIZE), PACK(DSIZE, 1));
PUT(heap_listp + (2 * WSIZE), PACK(DSIZE, 1));
PUT(heap_listp + (3 * WSIZE), PACK(0, 1));
heap_listp += (2 * WSIZE);
free_listp = NULL;

return 0;
}

/*
* mm_malloc - Allocate a block by incrementing the brk pointer.
* Always allocate a block whose size is a multiple of the alignment.
*/
void *mm_malloc(size_t size) {

size_t asize;
size_t extendsize;
char *bp;

if (size == 0) return NULL;

if (size <= DSIZE) asize = 2 * DSIZE;
else asize = DSIZE * ((size + (DSIZE) + (DSIZE - 1)) / DSIZE);

bp = (char *)find_fit(asize);

if (bp != NULL) {
place(bp, asize);
return bp;
}

extendsize = asize;

bp = (char *)extend_heap(extendsize / WSIZE);

if (bp == NULL) return NULL;

place(bp, asize);


return bp;
}

/*
* mm_free - Freeing a block does nothing.
*/
void mm_free(void *ptr)
{
if (ptr == NULL) return;

size_t size = GET_SIZE(HDRP(ptr));

PUT(HDRP(ptr), PACK(size, 0));
PUT(FTRP(ptr), PACK(size, 0));

SET_PREV(ptr, 0), SET_NEXT(ptr, 0);

coalesce(ptr);
}

/*
* mm_realloc - Implemented simply in terms of mm_malloc and mm_free
*/
void *mm_realloc(void *ptr, size_t size)
{
if (size == 0) {
mm_free(ptr);
return NULL;
}

if (ptr == NULL) {
return mm_malloc(size);
}

void *newptr = mm_malloc(size);
if (!newptr) return NULL;

size_t oldsize = GET_SIZE(HDRP(ptr));

if (size < oldsize) oldsize = size;

memcpy(newptr, ptr, oldsize);
mm_free(ptr);
return newptr;

}

void checkheap(int verbose);

static void printblock(void *bp)
{
size_t hsize, halloc, fsize, falloc;

checkheap(0);
hsize = GET_SIZE(HDRP(bp));
halloc = GET_ALLOC(HDRP(bp));
fsize = GET_SIZE(FTRP(bp));
falloc = GET_ALLOC(FTRP(bp));

if (hsize == 0) {
printf("%p: EOL\n", bp);
return;
}

printf("%p: header: [%ld:%c] footer: [%ld:%c]\n", bp,
hsize, (halloc ? 'a' : 'f'),
fsize, (falloc ? 'a' : 'f'));
}

static void checkblock(void *bp)
{
if ((size_t)bp % 8)
printf("Error: %p is not doubleword aligned\n", bp);
if (GET(HDRP(bp)) != GET(FTRP(bp)))
printf("Error: header does not match footer\n");
}

/*
* checkheap - Minimal check of the heap for consistency
*/
void checkheap(int verbose)
{
char *bp = heap_listp;

if (verbose)
printf("Heap (%p):\n", heap_listp);

if ((GET_SIZE(HDRP(heap_listp)) != DSIZE) || !GET_ALLOC(HDRP(heap_listp)))
printf("Bad prologue header\n");
checkblock(heap_listp);

for (bp = heap_listp; GET_SIZE(HDRP(bp)) > 0; bp = NEXT_BLKP(bp)) {
if (verbose)
printblock(bp);
checkblock(bp);
}

if (verbose)
printblock(bp);
if ((GET_SIZE(HDRP(bp)) != 0) || !(GET_ALLOC(HDRP(bp))))
printf("Bad epilogue header\n");
}

45 (util) + 40 (thru) = 85/100

优化的点主要在速度上