...
A similar situation arises when realloc()
is supplied a pointer to non-dynamically allocated memory. The realloc()
function is used to resize a block of dynamic memory. If realloc()
is supplied a pointer to memory not allocated by a memory allocation function, such as malloc()
, the program may also terminate abnormally.
Non-compliant Code Example 1
The following piece of code validates the number of command line arguemnts. If the correct number of commmand line arguments have been specified, then the requested amount of memory is validated to assure it is a acceptable size and the memory allocated with malloc()
. Next, the second command line argument is copied into str
for further processing. Once this processing is complete, str
is freed. However, if the incorrect number of arguments have been specified, str
is set to a string literal and printed. Because str
now references memory that was not dynamically allocated, an error will occur when str
memory is freed.
Code Block |
---|
#define MAX_SIZE_ALLOWED 1000 int main(int argc, char *argv[]) { char *str = NULL; size_t len; if (argc == 2) { len = strlen(argv[1])+1; if (len > MAX_SIZE_ALLOWED) { /* Handle Error */ } str = malloc(len); if (str == NULL) { /* Handle Allocation Error */ } strcpy(str,argv[1]); } else { str = "usage: $>a.exe [string]"; printf("%s\n", str); } /* ... */ free(str); return 0; } |
Compliant Solution 1
In the compliant solution, the program has been changed to eliminate the possibility of str
referencing non-dynamic memory when it is supplied to free()
.
Code Block |
---|
#define MAX_SIZE_ALLOWED 1000 int main(int argc, char *argv[]) { char *str = NULL; size_t len; if (argc == 2) { len = strlen(argv[1])+1; if (len > MAX_SIZE_ALLOWED) { /* Handle Error */ } str = malloc(len); if (str == NULL) { /* Handle Allocation Error */ } strcpy(str, argv[1]); } else { printf("usage: $>a.exe [string]\n"); return -1; } /* ... */ free(str); return 0; } |