hoi :) thanks Linus for your nice implementation. Your core code is so much nicer than my hacked-up prototype :-). I only had little time to actually have a look at it but the core is very similiar to my approach and I'll try to rebase some of my code on top of yours in the following days. The only thing I disagree with you is in using HEAD of the submodule: On Mon, Apr 09, 2007 at 09:20:29PM -0700, Linus Torvalds wrote: > +static int ce_compare_gitlink(struct cache_entry *ce) > +{ > + unsigned char sha1[20]; > + > + /* > + * We don't actually require that the .git directory > + * under DIRLNK directory be a valid git directory. It > + * might even be missing (in case nobody populated that > + * sub-project). > + * > + * If so, we consider it always to match. > + */ > + if (resolve_gitlink_ref(ce->name, "HEAD", sha1) < 0) > + return 0; > + return hashcmp(sha1, ce->sha1); > +} > @@ -2332,6 +2333,8 @@ int index_path(unsigned char *sha1, const char *path, struct stat *st, int write > path); > free(target); > break; > + case S_IFDIR: > + return resolve_gitlink_ref(path, "HEAD", sha1); > default: > return error("%s: unsupported file type", path); > } Always using HEAD of the submodule makes branches in the submodule useless. Whenever you do a checkout in the supermodule you also have to update the submodule and this update has to change the same thing which is read above. Updating the branch which HEAD points to is dangerous. You could overwrite some unrelated branch just because the user forgot to switch back to his supermodule-tracking-branch. The user would always have to make sure that all the submodules are in the correct state for an update of the supermodule. Updating HEAD directly is possible now and may make some sense, but you still get problems when you want to switch to some temporary branch in the submodule. You have no chance to get back to the original supermodule version and now your temporary submodule branch gets shown as the new submodule version which should be part of the supermodule. The submodule version which is stored in the supermodules tree is kind of a hidden/remote reference/branch. When working on a remote branch we first create a local working branch and then sync it with the remote one. I think that it makes sense to use the same model for submodules: have one local branch in the submodule which is used for all work that is done in the supermodule context. So my advice is: Always read and write one dedicated branch (hardcoded "master" or configurable) when the supermodule wants to access a submodule. Then you have two type of branches: You can branch the supermodule and have you own branch of the entire project with all submodules. Use this if you want to commit your work on the submodule into the supermodule. You can also branch the submodule to effectively disconnect the submodule from the supermodule temporarily. You can use this to do some experimental/debugging stuff which should not yet go into the supermodule. Once you want this branch to show up in the supermodule, just merge it to "master" and commit it to the supermodule (and now its in the supermodule branch, the submodule branch is not needed any more). See also the discussion about it in the messages around http://marc.info/?l=git&m=116636334226668&w=2 -- Martin Waitz
Attachment:
signature.asc
Description: Digital signature