CS 208 f21 — Dynamic Memory Allocation: malloc
and free
Table of Contents
1 Introduction
// static global data, size fixed at compile time, exists for the lifetime of the program int array[1024]; void foo(int n) { // stack-allocated data, known lifetime (deallocated on return) int tmp; int local_array[n]; // some versions of C allow dynamically-sized stack allocation // dynamic (heap) data, size and lifetime known only at runtime int* dyn = (int*) malloc(n * sizeof(int)); // good practices: // sizeof makes code more portable // void* is implicitly cast into any pointer type; explicit typecast will help you // catch coding errors when pointer types don’t match }
Two big questions we'll focus on for the next two weeks:
- how do we manage the scarce resource of physical memory while providing all processes as much virtual memory as they need?
- implemented by the operating system kernel and hardware
- how do we handle
malloc
andfree
quickly and efficiently?- implemented by the C library
Lab 4 will task you with doing 2., so we will start there.
2 Dynamic Memory Allocation
We don't always know how much space we will need for certain data structures, etc. ahead of time. Hardcoded sizes everywhere can lead to problems and become a maintenance nightmare. Hence, we wany the ability to allocate memory as we go along (dynamically)
2.1 Type of Allocators
- Allocator organizes heap as a collection of variable-sized blocks, which are either allocated or free
- explicit allocators: require manual requests and frees (
malloc
package in C) - implicit allocators: unused blocks are automatically detected and freed (garbage collection)
2.1.1 malloc
package
malloc
does no initialization, usecalloc
to request zero-initialized memory- returns pointer to the beginning of allocated block,
NULL
indicates failed request - typically 16-byte aligned on x86-64
- returns pointer to the beginning of allocated block,
- change the size of previously allocated block with
realloc
- underneath, allocation can use
mmap
andmunmap
or use thevoid *sbrk(intptr_t incr)
functionsbrk
grows or shrinks the heap by addingincr
to the kernel'sbrk
pointer, returns old value ofbrk
free
takes a pointer to the beginning of a block previously allocated bymalloc
,calloc
, orrealloc
- marks that memory as free and available for future allocations
- behavior undefined on other arguments, no indication of error
2.2 Spreadsheet Example
2.3 Allocator Goals and Requirements
2.3.1 Requirements
- handling arbitrary request sequences: cannot make any assumptions about the sequencing of allocate and free requests
- making immediate responses to requests: no reordering or batching of requests to improve performance
- using only the heap: allocator's internal data structures must also be stored on the heap
- aligning blocks: blocks need to meet alignment requirements for any type of data they might hold
- not modifying allocated blocks: cannot modify or move blocks when they are allocated
2.3.2 Goals
- maximizing throughput: maximize requests per second
- maximizing memory utilization: use the greatest possible fraction of the space allocated for the heap. Peak utilization metric is the maximum such fraction achieved over the course of \(n\) requests.
- these are in tension: respond to a request faster vs respond with a better choice of block
2.4 Fragmenataion
- internal fragmentation: allocated block is larger than the amount requested (due to minimum block size or alignment requirements)
- external fragmentation: when allocation/free pattern leaves "holes" between blocks
- symptom: there is enough total free memory to satisfy a request, but no single free block is large enough
- example: if final request on spreadsheet example were for 48 bytes
- difficult to quantify as it depends on future requests
2.5 Implementation Issues
- free block organization: how to keep track of them
- placement: which free block should we choose to fulfill a request
- splitting: what do we do with the remainder of a free block after part of it is allocated
- coalescing: what do we do when a block is freed
3 Quick Checks
Which of the following is NOT a source of internal fragmentation?1
- Placement policy (e.g., returning a larger block than requested)
- Padding for alignment purposes
- Pattern of future requests
- Memory added to blocks in order to maintain heap data structures
Which of the following statements is FALSE?2
- Temporary arrays should not be allocated on the heap
- malloc returns the address of a block that is filled with arbitrary data
- Memory utilization is affected by both internal and external fragmentation
- An allocation failure will cause your program to stop
4 Practice
Assume the heap starts at address 0x1000
with no allocated blocks. brk
is at 0x1100
. The requests below arrive in the order they are given. The system will allocate a 16-byte aligned block in the first available space after the start of the heap. For each request, give the block size and the address returned to the application.3
malloc(18)
malloc(24)
malloc(10)
malloc(32)
malloc(33)
Does the system need to call sbrk
at any point?
Footnotes:
The pattern of future requests is not a source of internal fragmentation
The false statement is an allocation failure will cause your program to stop.
When malloc
fails, it doesn't crash, but returns NULL.
It is the program's job to check this return value and act accordingly.
- Block size of 32, returns
0x1000
- Block size of 32, returns
0x1020
- Block size of 16, returns
0x1040
- Block size of 32, returns
0x1050
- Block size of 48, returns
0x1070
All of these fit below 0x1100
, so a call to sbrk
is not needed.