1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2008-2017 - TortoiseGit
5 // This program is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU General Public License
7 // as published by the Free Software Foundation; either version 2
8 // of the License, or (at your option) any later version.
10 // This program is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License
16 // along with this program; if not, write to the Free Software Foundation,
17 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 // gitdll.cpp : Defines the exported functions for the DLL application.
23 #include "../build/libgit-defines.h"
25 #pragma warning(disable: 4100 4018 4127 4244 4267)
26 #include "git-compat-util.h"
38 #include "run-command.h"
42 extern char g_last_error
[];
43 const char * g_prefix
;
45 extern void die_dll(const char *err
, va_list params
);
46 extern void handle_error(const char* err
, va_list params
);
47 extern void handle_warning(const char* warn
, va_list params
);
48 extern int die_is_recursing_dll(void);
50 extern void free_all_pack(void);
51 extern void reset_git_env(void);
52 extern void drop_all_attr_stacks(void);
53 extern void git_atexit_dispatch(void);
54 extern void git_atexit_clear(void);
55 extern void invalidate_ref_cache(void);
56 extern void cmd_log_init(int argc
, const char** argv
, const char* prefix
, struct rev_info
* rev
, struct setup_revision_opt
* opt
);
57 extern int estimate_commit_count(struct rev_info
* rev
, struct commit_list
* list
);
58 extern int log_tree_commit(struct rev_info
*, struct commit
*);
59 extern int write_entry(struct cache_entry
* ce
, char* path
, const struct checkout
* state
, int to_tempfile
);
60 extern struct object
* deref_tag(struct object
* o
, const char* warn
, int warnlen
);
61 extern void diff_flush_stat(struct diff_filepair
* p
, struct diff_options
* o
, struct diffstat_t
* diffstat
);
62 extern void free_diffstat_info(struct diffstat_t
* diffstat
);
63 extern int for_each_reflog_ent(const char* refname
, each_reflog_ent_fn fn
, void* cb_data
);
64 extern int for_each_ref_in(const char* prefix
, each_ref_fn fn
, void* cb_data
);
68 set_die_routine(die_dll
);
69 set_error_routine(handle_error
);
70 set_warn_routine(handle_warning
);
71 set_die_is_recursing_routine(die_is_recursing_dll
);
74 int git_get_sha1(const char *name
, GIT_HASH sha1
)
76 return get_sha1(name
,sha1
);
79 static int convert_slash(char * path
)
92 char path
[MAX_PATH
+1];
95 _setmode(_fileno(stdin
), _O_BINARY
);
96 _setmode(_fileno(stdout
), _O_BINARY
);
97 _setmode(_fileno(stderr
), _O_BINARY
);
99 GetModuleFileName(NULL
, path
, MAX_PATH
);
102 git_extract_argv0_path(path
);
104 // set HOME if not set already
105 gitsetenv("HOME", get_windows_home_directory(), 0);
106 drop_all_attr_stacks();
108 g_prefix
= setup_git_directory();
109 git_config(git_default_config
, NULL
);
110 invalidate_ref_cache();
115 static int git_parse_commit_author(struct GIT_COMMIT_AUTHOR
* author
, const char* pbuff
)
120 end
=strchr(pbuff
,'<');
125 author
->NameSize
= (int)(end
- pbuff
- 1);
128 end
= strchr(pbuff
, '>');
132 author
->Email
= pbuff
;
133 author
->EmailSize
= (int)(end
- pbuff
);
137 author
->Date
= atol(pbuff
);
138 end
= strchr(pbuff
, ' ');
143 author
->TimeZone
= atol(pbuff
);
148 int git_parse_commit(GIT_COMMIT
*commit
)
155 p
= (struct commit
*)commit
->m_pGitCommit
;
157 memcpy(commit
->m_hash
, p
->object
.oid
.hash
, GIT_HASH_SIZE
);
159 commit
->m_Encode
= NULL
;
160 commit
->m_EncodeSize
= 0;
162 commit
->buffer
= detach_commit_buffer(commit
->m_pGitCommit
, NULL
);
164 pbuf
= commit
->buffer
;
167 if (strncmp(pbuf
, "author", 6) == 0)
169 ret
= git_parse_commit_author(&commit
->m_Author
,pbuf
+ 7);
173 else if (strncmp(pbuf
, "committer", 9) == 0)
175 ret
= git_parse_commit_author(&commit
->m_Committer
,pbuf
+ 10);
179 pbuf
= strchr(pbuf
,'\n');
183 else if (strncmp(pbuf
, "encoding", 8) == 0)
186 commit
->m_Encode
=pbuf
;
187 end
= strchr(pbuf
,'\n');
188 commit
->m_EncodeSize
= (int)(end
-pbuf
);
191 // the headers end after the first empty line
192 else if (*pbuf
== '\n')
196 commit
->m_Subject
=pbuf
;
197 end
= strchr(pbuf
,'\n');
199 commit
->m_SubjectSize
= (int)strlen(pbuf
);
202 commit
->m_SubjectSize
= (int)(end
- pbuf
);
204 commit
->m_Body
= pbuf
;
205 commit
->m_BodySize
= (int)strlen(pbuf
);
210 pbuf
= strchr(pbuf
,'\n');
217 int git_get_commit_from_hash(GIT_COMMIT
* commit
, const GIT_HASH hash
)
226 memset(commit
,0,sizeof(GIT_COMMIT
));
228 commit
->m_pGitCommit
= p
= lookup_commit(hash
);
233 ret
= parse_commit(p
);
237 return git_parse_commit(commit
);
240 int git_get_commit_first_parent(GIT_COMMIT
*commit
,GIT_COMMIT_LIST
*list
)
242 struct commit
*p
= commit
->m_pGitCommit
;
247 *list
= (GIT_COMMIT_LIST
*)p
->parents
;
251 int git_commit_is_root(const GIT_COMMIT
* commit
)
253 struct commit
* p
= commit
->m_pGitCommit
;
254 return (struct commit_list
**)p
->parents
? 1 : 0;
257 int git_get_commit_next_parent(GIT_COMMIT_LIST
*list
, GIT_HASH hash
)
259 struct commit_list
*l
;
263 l
= *(struct commit_list
**)list
;
268 memcpy(hash
, l
->item
->object
.oid
.hash
, GIT_HASH_SIZE
);
270 *list
= (GIT_COMMIT_LIST
*)l
->next
;
276 int git_free_commit(GIT_COMMIT
*commit
)
278 struct commit
*p
= commit
->m_pGitCommit
;
281 free_commit_list(p
->parents
);
284 free_tree_buffer(p
->tree
);
286 #pragma warning(push)
287 #pragma warning(disable: 4090)
289 free(commit
->buffer
);
292 p
->object
.parsed
= 0;
296 memset(commit
,0,sizeof(GIT_COMMIT
));
300 char **strtoargv(char *arg
, int *size
)
322 argv
=malloc(strlen(arg
)+1 + (count
+2)*sizeof(void*));
323 p
=(char*)(argv
+count
+2);
342 if((*arg
== space
) || (*arg
== 0))
358 int git_open_log(GIT_LOG
* handle
, char * arg
)
360 struct rev_info
*p_Rev
;
363 struct setup_revision_opt opt
;
366 unsigned int obj_size
= get_max_object_index();
367 for (unsigned int i
= 0; i
< obj_size
; ++i
)
369 struct object
*ob
= get_indexed_object(i
);
373 if (ob
->parsed
&& ob
->type
== OBJ_COMMIT
)
375 struct commit
* commit
= (struct commit
*)ob
;
376 free_commit_list(commit
->parents
);
377 commit
->parents
= NULL
;
379 free_tree_buffer(commit
->tree
);
387 argv
= strtoargv(arg
,&argc
);
392 p_Rev
= malloc(sizeof(struct rev_info
));
399 memset(p_Rev
,0,sizeof(struct rev_info
));
401 invalidate_ref_cache();
403 init_revisions(p_Rev
, g_prefix
);
406 memset(&opt
, 0, sizeof(opt
));
409 cmd_log_init(argc
, argv
, g_prefix
,p_Rev
,&opt
);
411 p_Rev
->pPrivate
= argv
;
416 int git_get_log_firstcommit(GIT_LOG handle
)
418 return prepare_revision_walk(handle
);
421 int git_get_log_estimate_commit_count(GIT_LOG handle
)
423 struct rev_info
*p_Rev
;
424 p_Rev
=(struct rev_info
*)handle
;
426 return estimate_commit_count(p_Rev
, p_Rev
->commits
);
429 int git_get_log_nextcommit(GIT_LOG handle
, GIT_COMMIT
*commit
, int follow
)
436 memset(commit
, 0, sizeof(GIT_COMMIT
));
438 commit
->m_pGitCommit
= get_revision(handle
);
439 if( commit
->m_pGitCommit
== NULL
)
442 if (follow
&& !log_tree_commit(handle
, commit
->m_pGitCommit
))
444 commit
->m_ignore
= 1;
447 commit
->m_ignore
= 0;
449 ret
=git_parse_commit(commit
);
456 struct notes_tree
**display_notes_trees
;
457 int git_close_log(GIT_LOG handle
)
461 struct rev_info
*p_Rev
;
462 p_Rev
=(struct rev_info
*)handle
;
464 free(p_Rev
->pPrivate
);
469 if (display_notes_trees
)
470 free_notes(*display_notes_trees
);
471 display_notes_trees
= 0;
475 int git_open_diff(GIT_DIFF
*diff
, char * arg
)
477 struct rev_info
*p_Rev
;
482 argv
= strtoargv(arg
,&argc
);
484 p_Rev
= malloc(sizeof(struct rev_info
));
485 memset(p_Rev
,0,sizeof(struct rev_info
));
487 p_Rev
->pPrivate
= argv
;
488 *diff
= (GIT_DIFF
)p_Rev
;
490 init_revisions(p_Rev
, g_prefix
);
491 git_config(git_diff_basic_config
, NULL
); /* no "diff" UI options */
494 argc
= setup_revisions(argc
, argv
, p_Rev
, NULL
);
498 int git_close_diff(GIT_DIFF handle
)
500 git_diff_flush(handle
);
503 struct rev_info
*p_Rev
;
504 p_Rev
=(struct rev_info
*)handle
;
506 free(p_Rev
->pPrivate
);
511 int git_diff_flush(GIT_DIFF diff
)
513 struct diff_queue_struct
*q
= &diff_queued_diff
;
514 struct rev_info
*p_Rev
;
515 p_Rev
= (struct rev_info
*)diff
;
520 for (int i
= 0; i
< q
->nr
; ++i
)
521 diff_free_filepair(q
->queue
[i
]);
527 q
->nr
= q
->alloc
= 0;
530 if (p_Rev
->diffopt
.close_file
)
531 fclose(p_Rev
->diffopt
.file
);
533 free_diffstat_info(&p_Rev
->diffstat
);
537 int git_root_diff(GIT_DIFF diff
, GIT_HASH hash
,GIT_FILE
*file
, int *count
, int isstat
)
540 struct rev_info
*p_Rev
;
541 struct diff_queue_struct
*q
= &diff_queued_diff
;
543 p_Rev
= (struct rev_info
*)diff
;
545 ret
=diff_root_tree_sha1(hash
, "", &p_Rev
->diffopt
);
552 diffcore_std(&p_Rev
->diffopt
);
554 memset(&p_Rev
->diffstat
, 0, sizeof(struct diffstat_t
));
555 for (int i
= 0; i
< q
->nr
; ++i
) {
556 struct diff_filepair
*p
= q
->queue
[i
];
557 //if (check_pair_status(p))
558 diff_flush_stat(p
, &p_Rev
->diffopt
, &p_Rev
->diffstat
);
570 int git_do_diff(GIT_DIFF diff
, GIT_HASH hash1
, GIT_HASH hash2
, GIT_FILE
* file
, int *count
,int isstat
)
572 struct rev_info
*p_Rev
;
574 struct diff_queue_struct
*q
= &diff_queued_diff
;
576 p_Rev
= (struct rev_info
*)diff
;
578 ret
= diff_tree_sha1(hash1
,hash2
,"",&p_Rev
->diffopt
);
587 diffcore_std(&p_Rev
->diffopt
);
588 memset(&p_Rev
->diffstat
, 0, sizeof(struct diffstat_t
));
589 for (int i
= 0; i
< q
->nr
; ++i
) {
590 struct diff_filepair
*p
= q
->queue
[i
];
591 //if (check_pair_status(p))
592 diff_flush_stat(p
, &p_Rev
->diffopt
, &p_Rev
->diffstat
);
603 int git_get_diff_file(GIT_DIFF diff
, GIT_FILE file
, int i
, char** newname
, char** oldname
, int* IsDir
, int* status
, int* IsBin
, int* inc
, int* dec
)
605 struct diff_queue_struct
*q
= &diff_queued_diff
;
606 struct rev_info
*p_Rev
;
607 p_Rev
= (struct rev_info
*)diff
;
609 q
= (struct diff_queue_struct
*)file
;
615 assert(newname
&& oldname
&& status
&& IsDir
);
617 *newname
= q
->queue
[i
]->two
->path
;
618 *oldname
= q
->queue
[i
]->one
->path
;
619 *status
= q
->queue
[i
]->status
;
621 *IsDir
= (q
->queue
[i
]->one
->mode
& S_IFDIR
) == S_IFDIR
;
623 *IsDir
= (q
->queue
[i
]->two
->mode
& S_IFDIR
) == S_IFDIR
;
625 if (q
->queue
[i
]->one
->mode
&& q
->queue
[i
]->two
->mode
&& DIFF_PAIR_TYPE_CHANGED(q
->queue
[i
]))
628 if(p_Rev
->diffstat
.files
)
631 for (j
= 0; j
< p_Rev
->diffstat
.nr
; ++j
)
633 if(strcmp(*newname
,p_Rev
->diffstat
.files
[j
]->name
)==0)
636 if( j
== p_Rev
->diffstat
.nr
)
644 *IsBin
= p_Rev
->diffstat
.files
[j
]->is_binary
;
646 *inc
= (int)p_Rev
->diffstat
.files
[j
]->added
;
648 *dec
= (int)p_Rev
->diffstat
.files
[j
]->deleted
;
659 int git_add_exclude(const char *string
, const char *base
,
660 int baselen
, EXCLUDE_LIST which
, int lineno
)
662 add_exclude(string
, base
, baselen
, which
, lineno
);
666 int git_create_exclude_list(EXCLUDE_LIST
*which
)
668 *which
= malloc(sizeof(struct exclude_list
));
669 memset(*which
,0,sizeof(struct exclude_list
));
673 int git_free_exclude_list(EXCLUDE_LIST which
)
675 struct exclude_list
*p
= (struct exclude_list
*) which
;
677 for (int i
= 0; i
< p
->nr
; ++i
)
679 free(p
->excludes
[i
]);
686 int git_check_excluded_1(const char *pathname
,
687 int pathlen
, const char *basename
, int *dtype
,
688 EXCLUDE_LIST el
, int ignorecase
)
690 ignore_case
= ignorecase
;
691 return is_excluded_from_list(pathname
, pathlen
, basename
, dtype
, el
);
694 int git_get_notes(const GIT_HASH hash
, char** p_note
)
699 format_display_notes(hash
, &sb
, "utf-8", 1);
700 *p_note
= strbuf_detach(&sb
,&size
);
707 int (*fn
)(int, const char **, const char *);
711 #define RUN_SETUP (1<<0)
713 static struct cmd_struct commands
[] = {
714 { "notes", cmd_notes
, RUN_SETUP
},
715 { "update-index", cmd_update_index
, RUN_SETUP
},
718 int git_run_cmd(char *cmd
, char *arg
)
726 for (int i
= 0; i
< sizeof(commands
) / sizeof(struct cmd_struct
); ++i
)
728 if(strcmp(cmd
,commands
[i
].cmd
)==0)
732 argv
= strtoargv(arg
,&argc
);
734 ret
= commands
[i
].fn(argc
, argv
, NULL
);
750 void git_exit_cleanup(void)
752 git_atexit_dispatch();
756 int git_for_each_reflog_ent(const char *ref
, each_reflog_ent_fn fn
, void *cb_data
)
758 return for_each_reflog_ent(ref
,fn
,cb_data
);
761 static int update_some(const unsigned char* sha1
, struct strbuf
* base
,
762 const char *pathname
, unsigned mode
, int stage
, void *context
)
764 struct cache_entry
*ce
;
765 UNREFERENCED_PARAMETER(stage
);
767 ce
= (struct cache_entry
*)context
;
770 return READ_TREE_RECURSIVE
;
772 hashcpy(ce
->oid
.hash
, sha1
);
773 memcpy(ce
->name
, base
->buf
, base
->len
);
774 memcpy(ce
->name
+ base
->len
, pathname
, strlen(pathname
));
775 ce
->ce_flags
= create_ce_flags((unsigned int)(strlen(pathname
) + base
->len
));
776 ce
->ce_mode
= create_ce_mode(mode
);
781 int git_checkout_file(const char* ref
, const char* path
, char* outputpath
)
783 struct cache_entry
*ce
;
787 struct checkout state
;
788 struct pathspec pathspec
;
789 const char *matchbuf
[1];
790 ret
= get_sha1(ref
, sha1
);
794 reprepare_packed_git();
795 root
= parse_tree_indirect(sha1
);
803 ce
= xcalloc(1, cache_entry_size(strlen(path
)));
806 parse_pathspec(&pathspec
, PATHSPEC_ALL_MAGIC
, PATHSPEC_PREFER_CWD
, path
, matchbuf
);
807 pathspec
.items
[0].nowildcard_len
= pathspec
.items
[0].len
;
808 ret
= read_tree_recursive(root
, "", 0, 0, &pathspec
, update_some
, ce
);
809 clear_pathspec(&pathspec
);
817 memset(&state
, 0, sizeof(state
));
819 state
.refresh_cache
= 0;
821 ret
= write_entry(ce
, outputpath
, &state
, 0);
834 static int get_config(const char *key_
, const char *value_
, void *cb
)
836 struct config_buf
*buf
;
837 buf
=(struct config_buf
*)cb
;
838 if(strcmp(key_
, buf
->key
))
842 strncpy(buf
->buf
,value_
,buf
->size
);
856 // wchar_t wrapper for program_data_config()
857 const wchar_t* wget_program_data_config(void)
859 static const wchar_t *programdata_git_config
= NULL
;
860 wchar_t wpointer
[MAX_PATH
];
862 if (programdata_git_config
)
863 return programdata_git_config
;
865 if (xutftowcs_path(wpointer
, program_data_config()) < 0)
868 programdata_git_config
= _wcsdup(wpointer
);
870 return programdata_git_config
;
873 // wchar_t wrapper for git_etc_gitconfig()
874 const wchar_t *wget_msysgit_etc(void)
876 static const wchar_t *etc_gitconfig
= NULL
;
877 wchar_t wpointer
[MAX_PATH
];
880 return etc_gitconfig
;
882 if (xutftowcs_path(wpointer
, git_etc_gitconfig()) < 0)
885 etc_gitconfig
= _wcsdup(wpointer
);
887 return etc_gitconfig
;
890 int git_get_config(const char *key
, char *buffer
, int size
)
892 const char *home
, *system
, *programdata
;
893 struct config_buf buf
;
894 struct git_config_source config_source
= { 0 };
896 struct config_options opts
= { 0 };
897 opts
.respect_includes
= 1;
906 char* local
= git_pathdup("config");
907 config_source
.file
= local
;
908 git_config_with_options(get_config
, &buf
, &config_source
, &opts
);
914 home
= get_windows_home_directory();
917 char* global
= xstrdup(mkpath("%s/.gitconfig", home
));
920 config_source
.file
= global
;
921 git_config_with_options(get_config
, &buf
, &config_source
, &opts
);
926 char* globalxdg
= xstrdup(mkpath("%s/.config/git/config", home
));
929 config_source
.file
= globalxdg
;
930 git_config_with_options(get_config
, &buf
, &config_source
, &opts
);
937 system
= git_etc_gitconfig();
940 config_source
.file
= system
;
941 git_config_with_options(get_config
, &buf
, &config_source
, &opts
);
946 programdata
= git_program_data_config();
949 config_source
.file
= programdata
;
950 git_config_with_options(get_config
, &buf
, &config_source
, &opts
);
956 // taken from msysgit: compat/mingw.c
957 const char *get_windows_home_directory(void)
959 static const char *home_directory
= NULL
;
963 return home_directory
;
965 if ((tmp
= getenv("HOME")) != NULL
&& *tmp
)
967 home_directory
= _strdup(tmp
);
968 return home_directory
;
971 if ((tmp
= getenv("HOMEDRIVE")) != NULL
)
973 struct strbuf buf
= STRBUF_INIT
;
974 strbuf_addstr(&buf
, tmp
);
975 if ((tmp
= getenv("HOMEPATH")) != NULL
)
977 strbuf_addstr(&buf
, tmp
);
978 if (is_directory(buf
.buf
))
980 home_directory
= strbuf_detach(&buf
, NULL
);
981 return home_directory
;
984 strbuf_release(&buf
);
987 if ((tmp
= getenv("USERPROFILE")) != NULL
&& *tmp
)
988 home_directory
= _strdup(tmp
);
990 return home_directory
;
993 // wchar_t wrapper for get_windows_home_directory()
994 const wchar_t *wget_windows_home_directory(void)
996 static const wchar_t *home_directory
= NULL
;
997 wchar_t wpointer
[MAX_PATH
];
1000 return home_directory
;
1002 if (xutftowcs_path(wpointer
, get_windows_home_directory()) < 0)
1005 home_directory
= _wcsdup(wpointer
);
1007 return home_directory
;
1010 int get_set_config(const char *key
, const char *value
, CONFIG_TYPE type
)
1012 char * config_exclusive_filename
= NULL
;
1018 config_exclusive_filename
= git_pathdup("config");
1021 case CONFIG_XDGGLOBAL
:
1023 const char *home
= get_windows_home_directory();
1026 if (type
== CONFIG_GLOBAL
)
1027 config_exclusive_filename
= xstrdup(mkpath("%s/.gitconfig", home
));
1029 config_exclusive_filename
= xstrdup(mkpath("%s/.config/git/config", home
));
1035 if(!config_exclusive_filename
)
1038 ret
= git_config_set_multivar_in_file_gently(config_exclusive_filename
, key
, value
, NULL
, 0);
1039 free(config_exclusive_filename
);
1043 struct mailmap_info
{
1048 struct mailmap_entry
{
1049 /* name and email for the simple mail-only case */
1053 /* name and email for the complex mail and name matching case */
1054 struct string_list namemap
;
1057 int git_read_mailmap(GIT_MAILMAP
*mailmap
)
1059 struct string_list
*map
;
1066 if ((map
= (struct string_list
*)calloc(1, sizeof(struct string_list
))) == NULL
)
1069 if ((result
= read_mailmap(map
, NULL
)) != 0)
1088 int git_lookup_mailmap(GIT_MAILMAP mailmap
, const char** email1
, const char** name1
, const char* email2
, void* payload
, const char *(*author2_cb
)(void*))
1090 struct string_list
*map
;
1096 map
= (struct string_list
*)mailmap
;
1098 while (imax
>= imin
)
1100 int i
= imin
+ ((imax
- imin
) / 2);
1101 struct string_list_item
*si
= (struct string_list_item
*)&map
->items
[i
];
1102 struct mailmap_entry
*me
= (struct mailmap_entry
*)si
->util
;
1103 int comp
= map
->cmp(si
->string
, email2
);
1109 const char *author2
= author2_cb(payload
);
1110 for (unsigned int j
= 0; j
< me
->namemap
.nr
; ++j
)
1112 struct string_list_item
*sj
= (struct string_list_item
*)&me
->namemap
.items
[j
];
1113 struct mailmap_info
*mi
= (struct mailmap_info
*)sj
->util
;
1115 if (!map
->cmp(sj
->string
, author2
))
1118 *email1
= mi
->email
;
1127 *email1
= me
->email
;
1141 void git_free_mailmap(GIT_MAILMAP mailmap
)
1146 clear_mailmap((struct string_list
*)mailmap
);