-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsmm.c
348 lines (316 loc) · 12.2 KB
/
smm.c
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
/*
COMP3511 Fall 2024
PA3: Simplified Memory Management (smm)
Your name: CHEANG, Weng Io
Your ITSC email: wicheang @connect.ust.hk
Declaration:
I declare that I am not involved in plagiarism
I understand that both parties (i.e., students providing the codes and students copying the codes) will receive 0 marks.
*/
#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/mman.h> // use mmap, munmap system calls
// ==== About Heap Management in Per-process memory space =======
//
// Implementation notes:
// sbrk/brk is obselete (should not be used in future)
// mmap/munmap with some global constants/variables are used to define a heap segment
//
// Note: DO NOT MODIFY heap_start, heap_end, heap_current_break directly
// mm_sbrk is implemented to simulate the expected results of sbrk/brk system calls
// Use mm_sbrk(). It provides a similar sbrk() function to adjust heap_current_break
//
// Heap illustration:
// heap_end - heap_start = HEAP_SIZE bytes
//
// |-------------------| <------ heap_end (the upper limit of the heap)
// | |
// | |
// |-------------------| <------ heap_current_break (mm_sbrk(0) returns this address)
// | |
// | Heap in used |
// | |
// | |
// |-------------------| <------ heap_start (the lower limit of the heap)
const int HEAP_SIZE = 8000; // heap size in bytes
void *heap_start = NULL;
void *heap_end = NULL;
void *heap_current_break = NULL;
// Usage:
// mm_sbrk(0) returns the current heap break point
// if sz > 0, mm_sbrk(sz) moves up the current heap break point (i.e., enlarge the heap in used) and returns the previous break point
// if sz < 0, mm_sbrk(sz) moves down the current heap break point (i.e., shrink the heap in used) and returns the previous break point
void *mm_sbrk(int sz)
{
if (heap_start == NULL || heap_end == NULL || heap_current_break == NULL)
return MAP_FAILED; // error address: (void*) -1
if (sz == 0)
return heap_current_break;
// Note: sz is positive
if (sz > 0 && heap_current_break + sz <= heap_end)
{
void *ret = heap_current_break;
heap_current_break += sz;
return ret;
}
// Note: sz is negative
if (sz < 0 && heap_current_break + sz >= heap_start)
{
void *ret = heap_current_break;
heap_current_break += sz;
return ret;
}
return MAP_FAILED; // error address
}
// ==== End heap management =======
const int MAX_POINTERS = 26;
const int MAX_OPERATIONS = 100;
const char OPERATION_TYPE_MALLOC = 'M';
const char OPERATION_TYPE_FREE = 'F';
const char OPERATION_TYPE_COMBINE_NEARBY_FREE = 'C';
#define OPERATION_STR_MALLOC "malloc"
#define OPERATION_STR_FREE "free"
#define OPERATION_STR_COMBINE_NEARBY_FREE "combine_nearby_free"
const char META_DATA_STATUS_FREE = 'f';
const char META_DATA_STATUS_OCCUPIED = 'o';
// Data structure of MetaData
//
// The memory layout for this project assignment is:
//
// |--------------| <-- heap_current_break
// | Data N |
// |--------------|
// | MetaData N |
// |--------------|
// | ... |
// | ... |
// |--------------|
// | Data 1 |
// |--------------|
// | MetaData 1 |
// |--------------| <--- heap_start
struct
__attribute__((__packed__)) // compiler directive, avoid "gcc" padding bytes to struct
MetaData
{
size_t size; // 8 bytes (in 64-bit OS)
char status; // 1 byte ('f' or 'o')
};
// calculate the meta data size and store as a constant (exactly 9 bytes)
const size_t meta_data_size = sizeof(struct MetaData);
void mm_print()
{
void *cur_heap_break = mm_sbrk(0);
void *cur = heap_start;
int i = 1;
while (cur < cur_heap_break)
{
struct MetaData *md = (struct MetaData *)cur;
printf("Block %02d: [%s] size = %4ld %s\n",
i++, // block number - counting from bottom
(md->status == META_DATA_STATUS_FREE) ? "FREE" : "OCCP", // free or occupied
md->size,
md->size == 1 ? "byte" : "bytes"); // size, in term of bytes
// Advance to the next meta data
cur += meta_data_size + md->size;
}
}
//Helper Fucntion split: to split one memory block into two if possible
void split(struct MetaData* cur_metatdata, void*target, size_t target_size){
size_t remain = cur_metatdata->size - target_size;
//if the remaining size after split is 9bytes or fewer, then no split
if(remain<=meta_data_size) return;
//else, we can do split
cur_metatdata->size = target_size;
struct MetaData* new_metadata = target + target_size;
new_metadata->size = remain - meta_data_size;
new_metadata->status = META_DATA_STATUS_FREE;
return;
}
void *mm_malloc(size_t size)
{
// TODO: implement our own malloc function here
if(size !=0){
void *temp_start = heap_start;
void *temp_end = heap_current_break;
while(temp_start < temp_end){
struct MetaData *current_metadata = (struct MetaData *)temp_start;
size_t current_size = current_metadata->size;
if (current_metadata->status == META_DATA_STATUS_FREE && current_size >= size){
//found a fit size block
current_metadata->status = META_DATA_STATUS_OCCUPIED;
//target is the starting address of the memory block
void *target = (void *)current_metadata +meta_data_size;
split(current_metadata, target, size);
return target;
}
else temp_start += meta_data_size +current_metadata->size;
}
//out of while loop, meaning no fit size blocks can be found -> mm_sbrk()
struct MetaData* new_metadata = mm_sbrk(size + meta_data_size);
new_metadata->size = size;
new_metadata->status = META_DATA_STATUS_OCCUPIED;
return (void*)new_metadata+meta_data_size;
}
return NULL; // you should return a suitable address here
}
void mm_free(void *p)
{
// TODO: implement our own free function here
struct MetaData* cur_metadata = p-meta_data_size;
cur_metadata->status = META_DATA_STATUS_FREE;
}
void mm_combine_nearby_free()
{
// TODO: implement the algorithm to combine nearby free blocks
if (heap_start == heap_current_break) return; // if no blocks, then just return
void *start = heap_start;
void *end = heap_start;
while( start< heap_current_break && end < heap_current_break){
//if current metadata is free, then open a search to find the consecutive free blocks
struct MetaData* cur_metadata_start = (struct MetaData*)start;
// printf("Current address: %i \n", start);
if(cur_metadata_start->status == META_DATA_STATUS_FREE){
// printf("Current address: %i is free, starting a search \n", start);
end =start;
size_t total_memory_size =0;
size_t num_consecutive_blocks = 0;
struct MetaData* cur_metadata_end = cur_metadata_start;
while(end<heap_current_break && cur_metadata_end->status == META_DATA_STATUS_FREE){
// printf("checking address: %i \n", end);
total_memory_size+=cur_metadata_end->size;
num_consecutive_blocks++;
// printf("Total memory of current block: %i \n", total_memory_size);
end += meta_data_size + cur_metadata_end->size;
cur_metadata_end = (struct MetaData*)end;
}
// printf("Exit the search, starting next search at address: %i \n",end );
cur_metadata_start->size = total_memory_size + (num_consecutive_blocks -1)*meta_data_size;
start = end;
cur_metadata_start= (struct MetaData*) start;
continue;
}
else{
// printf("Current address %i is not free, going to next block \n", start);
start += meta_data_size + cur_metadata_start->size;
cur_metadata_start = (struct MetaData*) start;
}
}
}
int main()
{
char operation_types[MAX_OPERATIONS];
char pointer_chars[MAX_OPERATIONS];
int malloc_sizes[MAX_OPERATIONS];
int sz_operations;
int i, j;
// Assume there are at most 26 different malloc/free
// Here is the rule to map the block_name to pointers index
// a=>0, b=>1, ..., z=>25
void *pointers[MAX_POINTERS];
for (i = 0; i < MAX_POINTERS; i++)
pointers[i] = NULL;
char *target = NULL;
char command[30]; // malloc/free/combine_nearby_free
char block_name; // a-z
size_t block_size; // a non-negative integer
// Part 1: read and store the input
scanf("%d", &sz_operations); // read the number of operations
for (i = 0; i < sz_operations; i++)
{
scanf("%s", command);
if (strcmp(command, OPERATION_STR_MALLOC) == 0)
{
scanf(" %c %ld", &block_name, &block_size);
operation_types[i] = OPERATION_TYPE_MALLOC;
pointer_chars[i] = block_name;
malloc_sizes[i] = block_size;
}
else if (strcmp(command, OPERATION_STR_FREE) == 0)
{
scanf(" %c", &block_name);
operation_types[i] = OPERATION_TYPE_FREE;
pointer_chars[i] = block_name;
}
else if (strcmp(command, OPERATION_STR_COMBINE_NEARBY_FREE) == 0)
{
operation_types[i] = OPERATION_TYPE_COMBINE_NEARBY_FREE;
}
}
// Part 2: Allocate the HEAP_SIZE memory from OS and
// setup heap_start, heap_end, and heap_current_break pointers
// On success, heap_start points to the starting address of the heap region
// At the beginning, heap_current_break is pointing to heap_start (heap is used in 0 bytes)
heap_start = mmap(NULL, HEAP_SIZE, PROT_READ | PROT_WRITE,
MAP_SHARED | MAP_ANONYMOUS, -1, 0);
if (heap_start == MAP_FAILED)
{
printf("Error in creating heap using mmap\n");
exit(-1);
}
heap_current_break = heap_start;
heap_end = heap_start + HEAP_SIZE;
// printf("Heap start: %i", heap_start);
// Part 3: Do the simulation
for (i = 0; i < sz_operations; i++)
{
if (operation_types[i] == OPERATION_TYPE_MALLOC)
{
block_name = pointer_chars[i];
block_size = malloc_sizes[i];
if (pointers[block_name - 'a'] != NULL)
{
printf("=== %s %c %ld ===\n", OPERATION_STR_MALLOC, block_name, block_size);
printf("malloc Error: %c is pointing to some memory address\n", block_name);
}
else
{
target = mm_malloc(block_size);
if (target != NULL)
{
// This operation ensures that the returned pointer is correct
// As we only fill characters up to the block size,
// no meta data should be erased
for (j = 0; j < block_size; j++)
target[j] = ' ';
}
pointers[block_name - 'a'] = target;
printf("=== %s %c %ld ===\n", OPERATION_STR_MALLOC, block_name, block_size);
mm_print();
}
}
else if (operation_types[i] == OPERATION_TYPE_FREE)
{
block_name = pointer_chars[i];
if (pointers[block_name - 'a'] == NULL)
{
printf("=== %s %c ===\n", OPERATION_STR_FREE, block_name);
printf("free Error: %c is pointing to NULL\n", block_name);
}
else
{
mm_free(pointers[block_name - 'a']);
pointers[block_name - 'a'] = NULL;
printf("=== %s %c ===\n", OPERATION_STR_FREE, block_name);
mm_print();
}
}
else if (operation_types[i] == OPERATION_TYPE_COMBINE_NEARBY_FREE)
{
mm_combine_nearby_free();
printf("=== Combine nearby free blocks ===\n");
mm_print();
}
}
// Part 4: return HEAP_SIZE memory to the OS
if (munmap(heap_start, HEAP_SIZE))
{
// failure case
printf("Error in munmap\n");
exit(-1);
}
return 0;
}