|  | #include "cache.h" | 
|  | #include "quote.h" | 
|  | #include "exec_cmd.h" | 
|  | #include "strbuf.h" | 
|  |  | 
|  | static int do_generic_cmd(const char *me, char *arg) | 
|  | { | 
|  | const char *my_argv[4]; | 
|  |  | 
|  | setup_path(); | 
|  | if (!arg || !(arg = sq_dequote(arg))) | 
|  | die("bad argument"); | 
|  | if (prefixcmp(me, "git-")) | 
|  | die("bad command"); | 
|  |  | 
|  | my_argv[0] = me + 4; | 
|  | my_argv[1] = arg; | 
|  | my_argv[2] = NULL; | 
|  |  | 
|  | return execv_git_cmd(my_argv); | 
|  | } | 
|  |  | 
|  | static int do_cvs_cmd(const char *me, char *arg) | 
|  | { | 
|  | const char *cvsserver_argv[3] = { | 
|  | "cvsserver", "server", NULL | 
|  | }; | 
|  |  | 
|  | if (!arg || strcmp(arg, "server")) | 
|  | die("git-cvsserver only handles server: %s", arg); | 
|  |  | 
|  | setup_path(); | 
|  | return execv_git_cmd(cvsserver_argv); | 
|  | } | 
|  |  | 
|  |  | 
|  | static struct commands { | 
|  | const char *name; | 
|  | int (*exec)(const char *me, char *arg); | 
|  | } cmd_list[] = { | 
|  | { "git-receive-pack", do_generic_cmd }, | 
|  | { "git-upload-pack", do_generic_cmd }, | 
|  | { "git-upload-archive", do_generic_cmd }, | 
|  | { "cvs", do_cvs_cmd }, | 
|  | { NULL }, | 
|  | }; | 
|  |  | 
|  | int main(int argc, char **argv) | 
|  | { | 
|  | char *prog; | 
|  | struct commands *cmd; | 
|  | int devnull_fd; | 
|  |  | 
|  | /* | 
|  | * Always open file descriptors 0/1/2 to avoid clobbering files | 
|  | * in die().  It also avoids not messing up when the pipes are | 
|  | * dup'ed onto stdin/stdout/stderr in the child processes we spawn. | 
|  | */ | 
|  | devnull_fd = open("/dev/null", O_RDWR); | 
|  | while (devnull_fd >= 0 && devnull_fd <= 2) | 
|  | devnull_fd = dup(devnull_fd); | 
|  | if (devnull_fd == -1) | 
|  | die("opening /dev/null failed (%s)", strerror(errno)); | 
|  | close (devnull_fd); | 
|  |  | 
|  | /* | 
|  | * Special hack to pretend to be a CVS server | 
|  | */ | 
|  | if (argc == 2 && !strcmp(argv[1], "cvs server")) | 
|  | argv--; | 
|  |  | 
|  | /* | 
|  | * We do not accept anything but "-c" followed by "cmd arg", | 
|  | * where "cmd" is a very limited subset of git commands. | 
|  | */ | 
|  | else if (argc != 3 || strcmp(argv[1], "-c")) | 
|  | die("What do you think I am? A shell?"); | 
|  |  | 
|  | prog = argv[2]; | 
|  | if (!strncmp(prog, "git", 3) && isspace(prog[3])) | 
|  | /* Accept "git foo" as if the caller said "git-foo". */ | 
|  | prog[3] = '-'; | 
|  |  | 
|  | for (cmd = cmd_list ; cmd->name ; cmd++) { | 
|  | int len = strlen(cmd->name); | 
|  | char *arg; | 
|  | if (strncmp(cmd->name, prog, len)) | 
|  | continue; | 
|  | arg = NULL; | 
|  | switch (prog[len]) { | 
|  | case '\0': | 
|  | arg = NULL; | 
|  | break; | 
|  | case ' ': | 
|  | arg = prog + len + 1; | 
|  | break; | 
|  | default: | 
|  | continue; | 
|  | } | 
|  | exit(cmd->exec(cmd->name, arg)); | 
|  | } | 
|  | die("unrecognized command '%s'", prog); | 
|  | } |