Re: "git-diff -p :/anything" always segfaults

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




On Sun, 11 Mar 2007, Junio C Hamano wrote:
> >
> > "pop_most_recent_commit()" simply doesn't work that way. It *never* 
> > returns NULL. So having it as part of a while-loop was buggy to begin 
> > with, and you fixed the test, but the thing is, it should just look like
> >
> > 	while (list) {
> > 		struct commit *commit;
> >
> > 		commit = pop_most_recent_commit(&list, ONELINE_SEEN);
> > 		..
> >
> > and the "pop_most_recent_commit()" simply shouldn't be part of the 
> > conditional at all.
> 
> That's what I did in my tentative commit based on Jim's patch
> (except "commit" is also used to determine the return value from
> the function).

Your version is *also* buggy.

And the bug is exactly that return value.

You simply cannot do it that way. You do

	...
	commit = pop_most_recent_commit(&list, ONELINE_SEEN);
	if (!commit)
		break;
	...

which is totally broken and pointless, and more importantly, *exactly* 
because you use "commit" outside the loop, it's broken.

It will now return the last commit it ever saw, whether that commit 
matches the buffer or not. Because "commit" will *never* be set to NULL 
ever again after the initial assignment!

The fact is, if you don't do it the way I described:

	while (list) {
		struct commit *commit;

		commit = pop_most_recent_commit(&list, ONELINE_SEEN);
		...

you will almost certainly have a bug. Why? Because the return value of 
"pop_most_recent_commit()" simply *makes*no*sense* outside the loop. It's 
that simple. 

Outside the loop, you have to have a *different* return value, namely the 
one that matches. Which may be NULL. Something that 
pop_most_recent_commit() simply never returns!

So I repeat my suggestion:

 - either use the sequence above
 - or change the semantics of "pop_most_recent_commit()" to return NULL if 
   the list is empty.

Considering that there's now been two totally broken versions of this 
loop, with the exact same bug, I'm inclined to say that you should just 
change "pop_most_recent_commit()" instead, and then change the loop back 
to its original form of "while ((commit = pop_..) != NULL) {"

Otherwise, please apply something like this, which also fixes the return 
value. We should return -1 on error!

We should add a test with

	git log :/hjashjs

or similar. It's supposed to return an error ("fatal: ambiguous argument 
':/hjashjs': unknown revision or path not in the working tree.")

		Linus

 [ Same return value rules as for the kernel and for system calls, either:

    - 0/1 for "false/true", where _1_ is success, and 0 is "false".

    - negative/0 for "error/no-error", where _0_ is success, and negative 
      is a failure.

   but not some mixture with 1 being failure! ]
---
diff --git a/sha1_name.c b/sha1_name.c
index 6b8b67b..40dd2d1 100644
--- a/sha1_name.c
+++ b/sha1_name.c
@@ -602,10 +602,10 @@ static int handle_one_ref(const char *path,
  */
 
 #define ONELINE_SEEN (1u<<20)
-int get_sha1_oneline(const char *prefix, unsigned char *sha1)
+static int get_sha1_oneline(const char *prefix, unsigned char *sha1)
 {
 	struct commit_list *list = NULL, *backup = NULL, *l;
-	struct commit *commit = NULL;
+	int retval = -1;
 
 	if (prefix[0] == '!') {
 		if (prefix[1] != '!')
@@ -619,6 +619,7 @@ int get_sha1_oneline(const char *prefix, unsigned char *sha1)
 		commit_list_insert(l->item, &backup);
 	while (list) {
 		char *p;
+		struct commit *commit;
 
 		commit = pop_most_recent_commit(&list, ONELINE_SEEN);
 		if (!commit)
@@ -628,13 +629,14 @@ int get_sha1_oneline(const char *prefix, unsigned char *sha1)
 			continue;
 		if (!prefixcmp(p + 2, prefix)) {
 			hashcpy(sha1, commit->object.sha1);
+			retval = 0;
 			break;
 		}
 	}
 	free_commit_list(list);
 	for (l = backup; l; l = l->next)
 		clear_commit_marks(l->item, ONELINE_SEEN);
-	return commit == NULL;
+	return retval;
 }
 
 /*
-
To unsubscribe from this list: send the line "unsubscribe git" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Linux Kernel Development]     [Gcc Help]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [V4L]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]     [Fedora Users]