From: Christian Weisgerber Subject: Replace fparseln(3) with getline(3) To: gameoftrees@openbsd.org Date: Sat, 2 Jan 2021 20:07:47 +0100 replace fparseln(3) with getline(3), for better portability This also reduces the malloc/free calls and fixes a memory leak in get_modified_file_content_status(). ok? diff refs/heads/main refs/heads/getline blob - 2cd2c5fefcbd23e58b1bef2416f1cf6133544a8e blob + cd435b5446b903251e192c7dcaaea046c7f36d55 --- lib/reference.c +++ lib/reference.c @@ -359,9 +359,9 @@ open_packed_ref(struct got_reference **ref, FILE *f, c { const struct got_error *err = NULL; char *abs_refname; - char *line; - size_t len; - const char delim[3] = {'\0', '\0', '\0'}; + char *line = NULL; + size_t linesize = 0; + ssize_t linelen; int i, ref_is_absolute = (strncmp(refname, "refs/", 5) == 0); *ref = NULL; @@ -369,13 +369,15 @@ open_packed_ref(struct got_reference **ref, FILE *f, c if (ref_is_absolute) abs_refname = (char *)refname; do { - line = fparseln(f, &len, NULL, delim, 0); - if (line == NULL) { + linelen = getline(&line, &linesize, f); + if (linelen == -1) { if (feof(f)) break; err = got_ferror(f, GOT_ERR_BAD_REF_DATA); break; } + if (linelen > 0 && line[linelen-1] == '\n') + line[linelen-1] = '\0'; for (i = 0; i < nsubdirs; i++) { if (!ref_is_absolute && asprintf(&abs_refname, "refs/%s/%s", subdirs[i], @@ -387,10 +389,10 @@ open_packed_ref(struct got_reference **ref, FILE *f, c if (err || *ref != NULL) break; } - free(line); if (err) break; } while (*ref == NULL); + free(line); return err; } @@ -871,6 +873,7 @@ got_ref_list(struct got_reflist_head *refs, struct got char *packed_refs_path, *path_refs = NULL; char *abs_namespace = NULL; char *buf = NULL, *ondisk_ref_namespace = NULL; + char *line = NULL; FILE *f = NULL; struct got_reference *ref; struct got_reflist_entry *new; @@ -963,19 +966,19 @@ got_ref_list(struct got_reflist_head *refs, struct got f = fopen(packed_refs_path, "r"); free(packed_refs_path); if (f) { - char *line; - size_t len; - const char delim[3] = {'\0', '\0', '\0'}; + size_t linesize; + ssize_t linelen; for (;;) { - line = fparseln(f, &len, NULL, delim, 0); - if (line == NULL) { + linelen = getline(&line, &linesize, f); + if (linelen == -1) { if (feof(f)) break; err = got_ferror(f, GOT_ERR_BAD_REF_DATA); goto done; } + if (linelen > 0 && line[linelen-1] == '\n') + line[linelen-1] = '\0'; err = parse_packed_ref_line(&ref, NULL, line); - free(line); if (err) goto done; if (ref) { @@ -1001,6 +1004,7 @@ got_ref_list(struct got_reflist_head *refs, struct got done: free(abs_namespace); free(buf); + free(line); free(path_refs); if (f && fclose(f) != 0 && err == NULL) err = got_error_from_errno("fclose"); @@ -1174,7 +1178,7 @@ delete_packed_ref(struct got_reference *delref, struct const struct got_error *err = NULL, *unlock_err = NULL; struct got_lockfile *lf = NULL; FILE *f = NULL, *tmpf = NULL; - char *packed_refs_path, *tmppath = NULL; + char *line = NULL, *packed_refs_path, *tmppath = NULL; struct got_reflist_head refs; int found_delref = 0; @@ -1204,21 +1208,21 @@ delete_packed_ref(struct got_reference *delref, struct goto done; } for (;;) { - char *line; - size_t len; - const char delim[3] = {'\0', '\0', '\0'}; + size_t linesize; + ssize_t linelen; struct got_reference *ref; struct got_reflist_entry *new; - line = fparseln(f, &len, NULL, delim, 0); - if (line == NULL) { + linelen = getline(&line, &linesize, f); + if (linelen == -1) { if (feof(f)) break; err = got_ferror(f, GOT_ERR_BAD_REF_DATA); goto done; } + if (linelen > 0 && line[linelen-1] == '\n') + line[linelen-1] = '\0'; err = parse_packed_ref_line(&ref, NULL, line); - free(line); if (err) goto done; if (ref == NULL) @@ -1310,6 +1314,7 @@ done: } free(tmppath); free(packed_refs_path); + free(line); got_ref_list_free(&refs); return err ? err : unlock_err; } blob - 38c6dc6f1c93d31c2b51ea0514de0fbd09d5dfc3 blob + 5dfe5fbfdef7f0419e78c3302df7d292acfa2f2e --- lib/worktree.c +++ lib/worktree.c @@ -1599,13 +1599,13 @@ get_modified_file_content_status(unsigned char *status GOT_DIFF_CONFLICT_MARKER_END }; int i = 0; - char *line; - size_t len; - const char delim[3] = {'\0', '\0', '\0'}; + char *line = NULL; + size_t linesize = 0; + ssize_t linelen; while (*status == GOT_STATUS_MODIFY) { - line = fparseln(f, &len, NULL, delim, 0); - if (line == NULL) { + linelen = getline(&line, &linesize, f); + if (linelen == -1) { if (feof(f)) break; err = got_ferror(f, GOT_ERR_IO); @@ -1620,6 +1620,7 @@ get_modified_file_content_status(unsigned char *status i++; } } + free(line); return err; } -- Christian "naddy" Weisgerber naddy@mips.inka.de