git.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf
@ 2014-02-28  7:58 Faiz Kothari
  2014-02-28  9:15 ` Eric Sunshine
  0 siblings, 1 reply; 5+ messages in thread
From: Faiz Kothari @ 2014-02-28  7:58 UTC (permalink / raw)
  To: git; +Cc: mhagger, Faiz Kothari

Signed-off-by: Faiz Kothari <faiz.off93@gmail.com>

Notes:
    I finally got what's happening, and why the errors were caused.
    packname is supposed to contain the complete path to the .pack file.
    Packs are stored as /path/to/<SHA1>.pack which I overlooked earlier.
    After inspecting what is happening in pack-write.c:finish_tmp_packfile()
    which indirectly modifies packname by appending the SHA1 and ".pack" to packname
    This is happening in these code snippets:
    	char *end_of_name_prefix = strrchr(name_buffer, 0);
    
    and later
    	sprintf(end_of_name_prefix, "%s.pack", sha1_to_hex(sha1));
    
    name_buffer is packname.buf
    Using const for the first argument of pack-write.c:finish_tmp_packfile()
    doesnot raise any compile time warning or error and not any runtime errors,
    since the packname.buf is on heap and has extra space to which more char can be written.
    If this was not the case,
    	for e.g. passing a constant string and modifying it.
    	This will result in a segmentation fault.
---
 bulk-checkin.c |    8 +++++---
 pack-write.c   |    2 +-
 pack.h         |    2 +-
 3 files changed, 7 insertions(+), 5 deletions(-)

diff --git a/bulk-checkin.c b/bulk-checkin.c
index 118c625..bbdf1ec 100644
--- a/bulk-checkin.c
+++ b/bulk-checkin.c
@@ -23,7 +23,7 @@ static struct bulk_checkin_state {
 static void finish_bulk_checkin(struct bulk_checkin_state *state)
 {
 	unsigned char sha1[20];
-	char packname[PATH_MAX];
+	struct strbuf packname = STRBUF_INIT;
 	int i;
 
 	if (!state->f)
@@ -42,9 +42,10 @@ static void finish_bulk_checkin(struct bulk_checkin_state *state)
 					 state->offset);
 		close(fd);
 	}
+	strbuf_addf(&packname, "%s/pack/pack-", get_object_directory());
+	strbuf_grow(&packname, 40 + 5);
 
-	sprintf(packname, "%s/pack/pack-", get_object_directory());
-	finish_tmp_packfile(packname, state->pack_tmp_name,
+	finish_tmp_packfile(packname.buf, state->pack_tmp_name,
 			    state->written, state->nr_written,
 			    &state->pack_idx_opts, sha1);
 	for (i = 0; i < state->nr_written; i++)
@@ -53,6 +54,7 @@ static void finish_bulk_checkin(struct bulk_checkin_state *state)
 clear_exit:
 	free(state->written);
 	memset(state, 0, sizeof(*state));
+	strbuf_release(&packname);
 
 	/* Make objects we just wrote available to ourselves */
 	reprepare_packed_git();
diff --git a/pack-write.c b/pack-write.c
index 605d01b..ac38867 100644
--- a/pack-write.c
+++ b/pack-write.c
@@ -336,7 +336,7 @@ struct sha1file *create_tmp_packfile(char **pack_tmp_name)
 	return sha1fd(fd, *pack_tmp_name);
 }
 
-void finish_tmp_packfile(char *name_buffer,
+void finish_tmp_packfile(const char *name_buffer,
 			 const char *pack_tmp_name,
 			 struct pack_idx_entry **written_list,
 			 uint32_t nr_written,
diff --git a/pack.h b/pack.h
index 12d9516..3b9e033 100644
--- a/pack.h
+++ b/pack.h
@@ -91,6 +91,6 @@ extern int encode_in_pack_object_header(enum object_type, uintmax_t, unsigned ch
 extern int read_pack_header(int fd, struct pack_header *);
 
 extern struct sha1file *create_tmp_packfile(char **pack_tmp_name);
-extern void finish_tmp_packfile(char *name_buffer, const char *pack_tmp_name, struct pack_idx_entry **written_list, uint32_t nr_written, struct pack_idx_option *pack_idx_opts, unsigned char sha1[]);
+extern void finish_tmp_packfile(const char *name_buffer, const char *pack_tmp_name, struct pack_idx_entry **written_list, uint32_t nr_written, struct pack_idx_option *pack_idx_opts, unsigned char sha1[]);
 
 #endif
-- 
1.7.9.5

^ permalink raw reply related	[flat|nested] 5+ messages in thread

* Re: [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf
  2014-02-28  7:58 [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf Faiz Kothari
@ 2014-02-28  9:15 ` Eric Sunshine
  2014-02-28 18:27   ` Faiz Kothari
  0 siblings, 1 reply; 5+ messages in thread
From: Eric Sunshine @ 2014-02-28  9:15 UTC (permalink / raw)
  To: Faiz Kothari; +Cc: Git List, Michael Haggerty

On Fri, Feb 28, 2014 at 2:58 AM, Faiz Kothari <faiz.off93@gmail.com> wrote:
> Signed-off-by: Faiz Kothari <faiz.off93@gmail.com>
>
> Notes:
>     I finally got what's happening, and why the errors were caused.
>     packname is supposed to contain the complete path to the .pack file.
>     Packs are stored as /path/to/<SHA1>.pack which I overlooked earlier.
>     After inspecting what is happening in pack-write.c:finish_tmp_packfile()
>     which indirectly modifies packname by appending the SHA1 and ".pack" to packname
>     This is happening in these code snippets:
>         char *end_of_name_prefix = strrchr(name_buffer, 0);
>
>     and later
>         sprintf(end_of_name_prefix, "%s.pack", sha1_to_hex(sha1));
>
>     name_buffer is packname.buf
>     Using const for the first argument of pack-write.c:finish_tmp_packfile()
>     doesnot raise any compile time warning or error and not any runtime errors,
>     since the packname.buf is on heap and has extra space to which more char can be written.
>     If this was not the case,
>         for e.g. passing a constant string and modifying it.
>         This will result in a segmentation fault.
> ---

This notes section is important to the ongoing email discussion,
however, it should be placed below the "---" line so that it does not
become part of the recorded commit message when the patch is applied
via "git am".

>  bulk-checkin.c |    8 +++++---
>  pack-write.c   |    2 +-
>  pack.h         |    2 +-
>  3 files changed, 7 insertions(+), 5 deletions(-)
>
> diff --git a/bulk-checkin.c b/bulk-checkin.c
> index 118c625..bbdf1ec 100644
> --- a/bulk-checkin.c
> +++ b/bulk-checkin.c
> @@ -23,7 +23,7 @@ static struct bulk_checkin_state {
>  static void finish_bulk_checkin(struct bulk_checkin_state *state)
>  {
>         unsigned char sha1[20];
> -       char packname[PATH_MAX];
> +       struct strbuf packname = STRBUF_INIT;
>         int i;
>
>         if (!state->f)
> @@ -42,9 +42,10 @@ static void finish_bulk_checkin(struct bulk_checkin_state *state)
>                                          state->offset);
>                 close(fd);
>         }
> +       strbuf_addf(&packname, "%s/pack/pack-", get_object_directory());
> +       strbuf_grow(&packname, 40 + 5);

There are several problems with this. First, magic numbers 40 and 5
convey no meaning to the reader. At the very least, they should be
named constants or a comment should explain them. More seriously,
though, this code is fragile since it has far too intimate knowledge
of the inner workings of finish_tmp_packfile(). If the implementation
of finish_tmp_packfile() changes in the future such that it writes
more than 45 additional characters to the incoming buffer, this will
break.

Rather than coupling finish_bulk_checkin() and finish_tmp_packfile()
so tightly, consider finish_tmp_packfile() a black box which just
"does its job" and then propose ways to make things work without
finish_bulk_checkin() having to know how that job is done.

> -       sprintf(packname, "%s/pack/pack-", get_object_directory());
> -       finish_tmp_packfile(packname, state->pack_tmp_name,
> +       finish_tmp_packfile(packname.buf, state->pack_tmp_name,
>                             state->written, state->nr_written,
>                             &state->pack_idx_opts, sha1);
>         for (i = 0; i < state->nr_written; i++)
> diff --git a/pack-write.c b/pack-write.c
> index 605d01b..ac38867 100644
> --- a/pack-write.c
> +++ b/pack-write.c
> @@ -336,7 +336,7 @@ struct sha1file *create_tmp_packfile(char **pack_tmp_name)
>         return sha1fd(fd, *pack_tmp_name);
>  }
>
> -void finish_tmp_packfile(char *name_buffer,
> +void finish_tmp_packfile(const char *name_buffer,

This is misleading and fragile. By specifying 'const',
finish_tmp_packfile() promises not to modify the content of the
incoming name_buffer, yet it breaks this promise by modifying the
buffer through the non-const end_of_name_prefix variable (after
dropping the 'const' via strrchr()).

>                          const char *pack_tmp_name,
>                          struct pack_idx_entry **written_list,
>                          uint32_t nr_written,

^ permalink raw reply	[flat|nested] 5+ messages in thread

* Re: [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf
  2014-02-28  9:15 ` Eric Sunshine
@ 2014-02-28 18:27   ` Faiz Kothari
  2014-02-28 21:59     ` Eric Sunshine
  2014-03-01  1:34     ` He Sun
  0 siblings, 2 replies; 5+ messages in thread
From: Faiz Kothari @ 2014-02-28 18:27 UTC (permalink / raw)
  To: Eric Sunshine; +Cc: git

Hi,
Thanks for the suggestions and remarks.
I rewrote bulk-checkin.c:finish_bulk_checkin() using strbuf. But saw
that Sun He has already implemented the same way I have done.
Should I submit my implementation as a patch?

Secondly,
I tried implementing this WITHOUT changing the prototype of the
function pack-write.c:finish_tmp_packfile().

For this I detached the buffer from strbuf in finish_bulk_checkin()
using strbuf_detach() and passed it to finish_tmp_packfile().

Inside finish_tmp_packfile, I attached the same buffer to a local
struct strbuf using strbuf_attach().
Now the problem is, two of the arguments to strbuf_attach() are
'alloc' and 'len' which are private members of the struct strbuf.
But since I am just passing the detached buffer, the information of
alloc and len is lost which is required at the time of attaching.
I cannot think of any better way of using strbuf and NOT modify the
prototype of finish_tmp_packfile()

As a workaround, I can determine alloc = (strlen(buf) + 1) and len =
strlen(buf) but AFAIK this is not always true and may break.
Any suggestions?

Thanks.

On Fri, Feb 28, 2014 at 2:45 PM, Eric Sunshine <sunshine@sunshineco.com> wrote:
> On Fri, Feb 28, 2014 at 2:58 AM, Faiz Kothari <faiz.off93@gmail.com> wrote:
>> Signed-off-by: Faiz Kothari <faiz.off93@gmail.com>
>>
>> Notes:
>>     I finally got what's happening, and why the errors were caused.
>>     packname is supposed to contain the complete path to the .pack file.
>>     Packs are stored as /path/to/<SHA1>.pack which I overlooked earlier.
>>     After inspecting what is happening in pack-write.c:finish_tmp_packfile()
>>     which indirectly modifies packname by appending the SHA1 and ".pack" to packname
>>     This is happening in these code snippets:
>>         char *end_of_name_prefix = strrchr(name_buffer, 0);
>>
>>     and later
>>         sprintf(end_of_name_prefix, "%s.pack", sha1_to_hex(sha1));
>>
>>     name_buffer is packname.buf
>>     Using const for the first argument of pack-write.c:finish_tmp_packfile()
>>     doesnot raise any compile time warning or error and not any runtime errors,
>>     since the packname.buf is on heap and has extra space to which more char can be written.
>>     If this was not the case,
>>         for e.g. passing a constant string and modifying it.
>>         This will result in a segmentation fault.
>> ---
>
> This notes section is important to the ongoing email discussion,
> however, it should be placed below the "---" line so that it does not
> become part of the recorded commit message when the patch is applied
> via "git am".
>
>>  bulk-checkin.c |    8 +++++---
>>  pack-write.c   |    2 +-
>>  pack.h         |    2 +-
>>  3 files changed, 7 insertions(+), 5 deletions(-)
>>
>> diff --git a/bulk-checkin.c b/bulk-checkin.c
>> index 118c625..bbdf1ec 100644
>> --- a/bulk-checkin.c
>> +++ b/bulk-checkin.c
>> @@ -23,7 +23,7 @@ static struct bulk_checkin_state {
>>  static void finish_bulk_checkin(struct bulk_checkin_state *state)
>>  {
>>         unsigned char sha1[20];
>> -       char packname[PATH_MAX];
>> +       struct strbuf packname = STRBUF_INIT;
>>         int i;
>>
>>         if (!state->f)
>> @@ -42,9 +42,10 @@ static void finish_bulk_checkin(struct bulk_checkin_state *state)
>>                                          state->offset);
>>                 close(fd);
>>         }
>> +       strbuf_addf(&packname, "%s/pack/pack-", get_object_directory());
>> +       strbuf_grow(&packname, 40 + 5);
>
> There are several problems with this. First, magic numbers 40 and 5
> convey no meaning to the reader. At the very least, they should be
> named constants or a comment should explain them. More seriously,
> though, this code is fragile since it has far too intimate knowledge
> of the inner workings of finish_tmp_packfile(). If the implementation
> of finish_tmp_packfile() changes in the future such that it writes
> more than 45 additional characters to the incoming buffer, this will
> break.
>
> Rather than coupling finish_bulk_checkin() and finish_tmp_packfile()
> so tightly, consider finish_tmp_packfile() a black box which just
> "does its job" and then propose ways to make things work without
> finish_bulk_checkin() having to know how that job is done.
>
>> -       sprintf(packname, "%s/pack/pack-", get_object_directory());
>> -       finish_tmp_packfile(packname, state->pack_tmp_name,
>> +       finish_tmp_packfile(packname.buf, state->pack_tmp_name,
>>                             state->written, state->nr_written,
>>                             &state->pack_idx_opts, sha1);
>>         for (i = 0; i < state->nr_written; i++)
>> diff --git a/pack-write.c b/pack-write.c
>> index 605d01b..ac38867 100644
>> --- a/pack-write.c
>> +++ b/pack-write.c
>> @@ -336,7 +336,7 @@ struct sha1file *create_tmp_packfile(char **pack_tmp_name)
>>         return sha1fd(fd, *pack_tmp_name);
>>  }
>>
>> -void finish_tmp_packfile(char *name_buffer,
>> +void finish_tmp_packfile(const char *name_buffer,
>
> This is misleading and fragile. By specifying 'const',
> finish_tmp_packfile() promises not to modify the content of the
> incoming name_buffer, yet it breaks this promise by modifying the
> buffer through the non-const end_of_name_prefix variable (after
> dropping the 'const' via strrchr()).
>
>>                          const char *pack_tmp_name,
>>                          struct pack_idx_entry **written_list,
>>                          uint32_t nr_written,

^ permalink raw reply	[flat|nested] 5+ messages in thread

* Re: [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf
  2014-02-28 18:27   ` Faiz Kothari
@ 2014-02-28 21:59     ` Eric Sunshine
  2014-03-01  1:34     ` He Sun
  1 sibling, 0 replies; 5+ messages in thread
From: Eric Sunshine @ 2014-02-28 21:59 UTC (permalink / raw)
  To: Faiz Kothari; +Cc: Git List

On Fri, Feb 28, 2014 at 1:27 PM, Faiz Kothari <faiz.off93@gmail.com> wrote:
> Thanks for the suggestions and remarks.

[Administrivia: On this list, top-posting is frowned upon; inline
responses are preferred.]

> I rewrote bulk-checkin.c:finish_bulk_checkin() using strbuf. But saw
> that Sun He has already implemented the same way I have done.
> Should I submit my implementation as a patch?

Yes. The purpose of these micro-projects is to expose you to the Git
project's development process so that you know what will be expected
of you as a GSoC student, and to give the GSoC mentors an opportunity
to evaluate your abilities and observe how you interact with the
reviewers.

> Secondly,
> I tried implementing this WITHOUT changing the prototype of the
> function pack-write.c:finish_tmp_packfile().
>
> For this I detached the buffer from strbuf in finish_bulk_checkin()
> using strbuf_detach() and passed it to finish_tmp_packfile().
>
> Inside finish_tmp_packfile, I attached the same buffer to a local
> struct strbuf using strbuf_attach().
> Now the problem is, two of the arguments to strbuf_attach() are
> 'alloc' and 'len' which are private members of the struct strbuf.
> But since I am just passing the detached buffer, the information of
> alloc and len is lost which is required at the time of attaching.
> I cannot think of any better way of using strbuf and NOT modify the
> prototype of finish_tmp_packfile()
>
> As a workaround, I can determine alloc = (strlen(buf) + 1) and len =
> strlen(buf) but AFAIK this is not always true and may break.
> Any suggestions?

That's getting rather convoluted. You may want to ask yourself if it
is really necessary for finish_tmp_packfile() to modify the buffer
passed in by the caller or if finish_pack_file() should instead take
responsibility for itself by allocating its own buffer (strbuf) in
which to do path manipulation.

^ permalink raw reply	[flat|nested] 5+ messages in thread

* Re: [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf
  2014-02-28 18:27   ` Faiz Kothari
  2014-02-28 21:59     ` Eric Sunshine
@ 2014-03-01  1:34     ` He Sun
  1 sibling, 0 replies; 5+ messages in thread
From: He Sun @ 2014-03-01  1:34 UTC (permalink / raw)
  To: Faiz Kothari, git

2014-03-01 2:27 GMT+08:00 Faiz Kothari <faiz.off93@gmail.com>:
> Hi,
> Thanks for the suggestions and remarks.
> I rewrote bulk-checkin.c:finish_bulk_checkin() using strbuf. But saw
> that Sun He has already implemented the same way I have done.
> Should I submit my implementation as a patch?
>
> Secondly,
> I tried implementing this WITHOUT changing the prototype of the
> function pack-write.c:finish_tmp_packfile().
>

I used to think the same as you.

> For this I detached the buffer from strbuf in finish_bulk_checkin()
> using strbuf_detach() and passed it to finish_tmp_packfile().
>
> Inside finish_tmp_packfile, I attached the same buffer to a local
> struct strbuf using strbuf_attach().
> Now the problem is, two of the arguments to strbuf_attach() are
> 'alloc' and 'len' which are private members of the struct strbuf.
> But since I am just passing the detached buffer, the information of
> alloc and len is lost which is required at the time of attaching.

One stupid solution may be that, alloc 8 byte that attached to strbuf's buf,
and fill in the len and alloc. We can take them out in finish_tmp_packfile.

But this may cause potential problems, that functions should have as rare
relationship as they could.
The goal of limiting the changes in one function is for this purpose.

So may be change the first paramater of finish_tmp_packfile is the best way
to deal with this situation.

> I cannot think of any better way of using strbuf and NOT modify the
> prototype of finish_tmp_packfile()
>
> As a workaround, I can determine alloc = (strlen(buf) + 1) and len =
> strlen(buf) but AFAIK this is not always true and may break.
> Any suggestions?
>
> Thanks.
>
> On Fri, Feb 28, 2014 at 2:45 PM, Eric Sunshine <sunshine@sunshineco.com> wrote:
>> On Fri, Feb 28, 2014 at 2:58 AM, Faiz Kothari <faiz.off93@gmail.com> wrote:
>>> Signed-off-by: Faiz Kothari <faiz.off93@gmail.com>
>>>
>>> Notes:
>>>     I finally got what's happening, and why the errors were caused.
>>>     packname is supposed to contain the complete path to the .pack file.
>>>     Packs are stored as /path/to/<SHA1>.pack which I overlooked earlier.
>>>     After inspecting what is happening in pack-write.c:finish_tmp_packfile()
>>>     which indirectly modifies packname by appending the SHA1 and ".pack" to packname
>>>     This is happening in these code snippets:
>>>         char *end_of_name_prefix = strrchr(name_buffer, 0);
>>>
>>>     and later
>>>         sprintf(end_of_name_prefix, "%s.pack", sha1_to_hex(sha1));
>>>
>>>     name_buffer is packname.buf
>>>     Using const for the first argument of pack-write.c:finish_tmp_packfile()
>>>     doesnot raise any compile time warning or error and not any runtime errors,
>>>     since the packname.buf is on heap and has extra space to which more char can be written.
>>>     If this was not the case,
>>>         for e.g. passing a constant string and modifying it.
>>>         This will result in a segmentation fault.
>>> ---
>>
>> This notes section is important to the ongoing email discussion,
>> however, it should be placed below the "---" line so that it does not
>> become part of the recorded commit message when the patch is applied
>> via "git am".
>>
>>>  bulk-checkin.c |    8 +++++---
>>>  pack-write.c   |    2 +-
>>>  pack.h         |    2 +-
>>>  3 files changed, 7 insertions(+), 5 deletions(-)
>>>
>>> diff --git a/bulk-checkin.c b/bulk-checkin.c
>>> index 118c625..bbdf1ec 100644
>>> --- a/bulk-checkin.c
>>> +++ b/bulk-checkin.c
>>> @@ -23,7 +23,7 @@ static struct bulk_checkin_state {
>>>  static void finish_bulk_checkin(struct bulk_checkin_state *state)
>>>  {
>>>         unsigned char sha1[20];
>>> -       char packname[PATH_MAX];
>>> +       struct strbuf packname = STRBUF_INIT;
>>>         int i;
>>>
>>>         if (!state->f)
>>> @@ -42,9 +42,10 @@ static void finish_bulk_checkin(struct bulk_checkin_state *state)
>>>                                          state->offset);
>>>                 close(fd);
>>>         }
>>> +       strbuf_addf(&packname, "%s/pack/pack-", get_object_directory());
>>> +       strbuf_grow(&packname, 40 + 5);
>>
>> There are several problems with this. First, magic numbers 40 and 5
>> convey no meaning to the reader. At the very least, they should be
>> named constants or a comment should explain them. More seriously,
>> though, this code is fragile since it has far too intimate knowledge
>> of the inner workings of finish_tmp_packfile(). If the implementation
>> of finish_tmp_packfile() changes in the future such that it writes
>> more than 45 additional characters to the incoming buffer, this will
>> break.
>>
>> Rather than coupling finish_bulk_checkin() and finish_tmp_packfile()
>> so tightly, consider finish_tmp_packfile() a black box which just
>> "does its job" and then propose ways to make things work without
>> finish_bulk_checkin() having to know how that job is done.
>>
>>> -       sprintf(packname, "%s/pack/pack-", get_object_directory());
>>> -       finish_tmp_packfile(packname, state->pack_tmp_name,
>>> +       finish_tmp_packfile(packname.buf, state->pack_tmp_name,
>>>                             state->written, state->nr_written,
>>>                             &state->pack_idx_opts, sha1);
>>>         for (i = 0; i < state->nr_written; i++)
>>> diff --git a/pack-write.c b/pack-write.c
>>> index 605d01b..ac38867 100644
>>> --- a/pack-write.c
>>> +++ b/pack-write.c
>>> @@ -336,7 +336,7 @@ struct sha1file *create_tmp_packfile(char **pack_tmp_name)
>>>         return sha1fd(fd, *pack_tmp_name);
>>>  }
>>>
>>> -void finish_tmp_packfile(char *name_buffer,
>>> +void finish_tmp_packfile(const char *name_buffer,
>>
>> This is misleading and fragile. By specifying 'const',
>> finish_tmp_packfile() promises not to modify the content of the
>> incoming name_buffer, yet it breaks this promise by modifying the
>> buffer through the non-const end_of_name_prefix variable (after
>> dropping the 'const' via strrchr()).
>>
>>>                          const char *pack_tmp_name,
>>>                          struct pack_idx_entry **written_list,
>>>                          uint32_t nr_written,
> --
> To unsubscribe from this list: send the line "unsubscribe git" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html
Cheers,
He Sun

^ permalink raw reply	[flat|nested] 5+ messages in thread

end of thread, other threads:[~2014-03-01  1:34 UTC | newest]

Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-02-28  7:58 [PATCH] rewrite bulk-checkin.c:finish_bulk_checkin() using strbuf Faiz Kothari
2014-02-28  9:15 ` Eric Sunshine
2014-02-28 18:27   ` Faiz Kothari
2014-02-28 21:59     ` Eric Sunshine
2014-03-01  1:34     ` He Sun

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).