My problem appears to lie in this function. I have tried returning a string, but have heard that mallocing some space is bad unless you explicitly free it later.
Explicitly freeing dynamically allocated memory is required to avoid memory leaks. However, it is not necessarily a task that the end users need to perform directly: an API often provides a function to deal with this.
In your case, you should provide a deinit_semver
function that does the clean up of memory that init_semver
has allocated dynamically. These two functions behave in a way that is similar to constructor and destructor; init_semver
is not a factory function, because it expects the semver
struct
to be allocated, rather than allocating it internally.
Here is one way of doing it:
void init_semver(semver * s, int major, int minor, int pathc, const char * note) {
s->major = major;
s->minor = minor;
s->patch = pathc;
size_t len = strlen(note);
s->note = malloc(len+1);
strcpy(s->note, note);
s->tag = malloc(40 + len);
sprintf(s->tag, "v%d.%d.%d-%s", major, minor, patch, note);
}
void deinit_semver(semver *s) {
free(s->note);
free(s->tag);
}
Note the changes above: rather than using fixed values for the components of struct
semver
, this code takes the values as parameters. In addition, the code copies the note
into a dynamically allocated buffer, rather than pointing to it directly.
The deinit
function does the clean-up by free
-ing both fields that were allocated dynamically.