1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2008-2013 - 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 "git-compat-util.h"
36 #include "run-command.h"
41 // This is an example of an exported variable
42 GITDLL_API
int ngitdll
=0;
44 // This is an example of an exported function.
45 GITDLL_API
int fngitdll(void)
50 // This is the constructor of a class that has been exported.
51 // see gitdll.h for the class definition
58 extern char g_last_error
[];
61 char * get_git_last_error()
66 extern void die_dll(const char *err
, va_list params
);
70 set_die_routine(die_dll
);
73 int git_get_sha1(const char *name
, GIT_HASH sha1
)
75 return get_sha1(name
,sha1
);
78 static int convert_slash(char * path
)
91 char path
[MAX_PATH
+1];
97 _setmode(_fileno(stdin
), _O_BINARY
);
98 _setmode(_fileno(stdout
), _O_BINARY
);
99 _setmode(_fileno(stderr
), _O_BINARY
);
101 // set HOME if not set already
102 getenv_s(&homesize
, NULL
, 0, "HOME");
105 _wputenv_s(L
"HOME", wget_windows_home_directory());
107 GetModuleFileName(NULL
, path
, MAX_PATH
);
110 git_extract_argv0_path(path
);
111 g_prefix
= prefix
= setup_git_directory();
112 ret
= git_config(git_default_config
, NULL
);
116 _putenv_s("HOME","");/* clear home evironment to avoid affact third part software*/
122 static int git_parse_commit_author(struct GIT_COMMIT_AUTHOR
*author
, char *pbuff
)
127 end
=strchr(pbuff
,'<');
132 author
->NameSize
= end
- pbuff
- 1;
135 end
= strchr(pbuff
, '>');
139 author
->Email
= pbuff
;
140 author
->EmailSize
= end
- pbuff
;
144 author
->Date
= atol(pbuff
);
145 end
= strchr(pbuff
, ' ');
150 author
->TimeZone
= atol(pbuff
);
155 int git_parse_commit(GIT_COMMIT
*commit
)
162 p
= (struct commit
*)commit
->m_pGitCommit
;
164 memcpy(commit
->m_hash
,p
->object
.sha1
,GIT_HASH_SIZE
);
166 commit
->m_Encode
= NULL
;
167 commit
->m_EncodeSize
= 0;
169 if(p
->buffer
== NULL
)
175 if (strncmp(pbuf
, "author", 6) == 0)
177 ret
= git_parse_commit_author(&commit
->m_Author
,pbuf
+ 7);
181 else if (strncmp(pbuf
, "committer", 9) == 0)
183 ret
= git_parse_commit_author(&commit
->m_Committer
,pbuf
+ 10);
187 pbuf
= strchr(pbuf
,'\n');
191 else if (strncmp(pbuf
, "encoding", 8) == 0)
194 commit
->m_Encode
=pbuf
;
195 end
= strchr(pbuf
,'\n');
196 commit
->m_EncodeSize
=end
-pbuf
;
199 // the headers end after the first empty line
200 else if (*pbuf
== '\n')
204 commit
->m_Subject
=pbuf
;
205 end
= strchr(pbuf
,'\n');
207 commit
->m_SubjectSize
= strlen(pbuf
);
210 commit
->m_SubjectSize
= end
- pbuf
;
212 commit
->m_Body
= pbuf
;
213 commit
->m_BodySize
= strlen(pbuf
);
218 pbuf
= strchr(pbuf
,'\n');
225 int git_get_commit_from_hash(GIT_COMMIT
*commit
, GIT_HASH hash
)
231 memset(commit
,0,sizeof(GIT_COMMIT
));
233 commit
->m_pGitCommit
= p
= lookup_commit(hash
);
241 ret
= parse_commit(p
);
245 return git_parse_commit(commit
);
248 int git_get_commit_first_parent(GIT_COMMIT
*commit
,GIT_COMMIT_LIST
*list
)
250 struct commit
*p
= commit
->m_pGitCommit
;
255 *list
= (GIT_COMMIT_LIST
*)p
->parents
;
258 int git_get_commit_next_parent(GIT_COMMIT_LIST
*list
, GIT_HASH hash
)
260 struct commit_list
*l
= *(struct commit_list
**)list
;
261 if(list
== NULL
|| l
==NULL
)
265 memcpy(hash
, l
->item
->object
.sha1
, GIT_HASH_SIZE
);
267 *list
= (GIT_COMMIT_LIST
*)l
->next
;
273 int git_free_commit(GIT_COMMIT
*commit
)
275 struct commit
*p
= commit
->m_pGitCommit
;
278 free_commit_list(p
->parents
);
288 memset(commit
,0,sizeof(GIT_COMMIT
));
292 char **strtoargv(char *arg
, int *size
)
314 argv
=malloc(strlen(arg
)+1 + (count
+2)*sizeof(void*));
315 p
=(char*)(argv
+count
+2);
334 if((*arg
== space
) || (*arg
== 0))
350 int git_open_log(GIT_LOG
* handle
, char * arg
)
352 struct rev_info
*p_Rev
;
356 struct setup_revision_opt opt
;
359 unsigned int obj_size
= get_max_object_index();
360 for(i
=0; i
<obj_size
; i
++)
362 struct object
*ob
= get_indexed_object(i
);
368 argv
= strtoargv(arg
,&argc
);
370 p_Rev
= malloc(sizeof(struct rev_info
));
371 memset(p_Rev
,0,sizeof(struct rev_info
));
376 invalidate_ref_cache(NULL
);
378 init_revisions(p_Rev
, g_prefix
);
381 memset(&opt
, 0, sizeof(opt
));
384 cmd_log_init(argc
, argv
, g_prefix
,p_Rev
,&opt
);
386 p_Rev
->pPrivate
= argv
;
391 int git_get_log_firstcommit(GIT_LOG handle
)
393 return prepare_revision_walk(handle
);
396 int git_get_log_estimate_commit_count(GIT_LOG handle
)
398 struct rev_info
*p_Rev
;
399 p_Rev
=(struct rev_info
*)handle
;
401 return estimate_commit_count(p_Rev
, p_Rev
->commits
);
404 int git_get_log_nextcommit(GIT_LOG handle
, GIT_COMMIT
*commit
, int follow
)
411 memset(commit
, 0, sizeof(GIT_COMMIT
));
413 commit
->m_pGitCommit
= get_revision(handle
);
414 if( commit
->m_pGitCommit
== NULL
)
417 if (follow
&& !log_tree_commit(handle
, commit
->m_pGitCommit
))
419 commit
->m_ignore
= 1;
422 commit
->m_ignore
= 0;
424 ret
=git_parse_commit(commit
);
431 struct notes_tree
**display_notes_trees
;
432 int git_close_log(GIT_LOG handle
)
436 struct rev_info
*p_Rev
;
437 p_Rev
=(struct rev_info
*)handle
;
439 free(p_Rev
->pPrivate
);
444 if (display_notes_trees
)
445 free_notes(*display_notes_trees
);
446 display_notes_trees
= 0;
450 int git_open_diff(GIT_DIFF
*diff
, char * arg
)
452 struct rev_info
*p_Rev
;
457 argv
= strtoargv(arg
,&argc
);
459 p_Rev
= malloc(sizeof(struct rev_info
));
460 memset(p_Rev
,0,sizeof(struct rev_info
));
462 p_Rev
->pPrivate
= argv
;
463 *diff
= (GIT_DIFF
)p_Rev
;
465 init_revisions(p_Rev
, g_prefix
);
466 git_config(git_diff_basic_config
, NULL
); /* no "diff" UI options */
469 argc
= setup_revisions(argc
, argv
, p_Rev
, NULL
);
473 int git_close_diff(GIT_DIFF handle
)
475 git_diff_flush(handle
);
478 struct rev_info
*p_Rev
;
479 p_Rev
=(struct rev_info
*)handle
;
481 free(p_Rev
->pPrivate
);
486 int git_diff_flush(GIT_DIFF diff
)
488 struct diff_queue_struct
*q
= &diff_queued_diff
;
489 struct rev_info
*p_Rev
;
491 p_Rev
= (struct rev_info
*)diff
;
496 for (i
= 0; i
< q
->nr
; i
++)
497 diff_free_filepair(q
->queue
[i
]);
503 q
->nr
= q
->alloc
= 0;
506 if (p_Rev
->diffopt
.close_file
)
507 fclose(p_Rev
->diffopt
.file
);
509 free_diffstat_info(&p_Rev
->diffstat
);
513 int git_root_diff(GIT_DIFF diff
, GIT_HASH hash
,GIT_FILE
*file
, int *count
, int isstat
)
516 struct rev_info
*p_Rev
;
518 struct diff_queue_struct
*q
= &diff_queued_diff
;
520 p_Rev
= (struct rev_info
*)diff
;
522 ret
=diff_root_tree_sha1(hash
, "", &p_Rev
->diffopt
);
529 diffcore_std(&p_Rev
->diffopt
);
531 memset(&p_Rev
->diffstat
, 0, sizeof(struct diffstat_t
));
532 for (i
= 0; i
< q
->nr
; i
++) {
533 struct diff_filepair
*p
= q
->queue
[i
];
534 //if (check_pair_status(p))
535 diff_flush_stat(p
, &p_Rev
->diffopt
, &p_Rev
->diffstat
);
546 int git_diff(GIT_DIFF diff
, GIT_HASH hash1
, GIT_HASH hash2
, GIT_FILE
* file
, int *count
,int isstat
)
548 struct rev_info
*p_Rev
;
551 struct diff_queue_struct
*q
= &diff_queued_diff
;
553 p_Rev
= (struct rev_info
*)diff
;
555 ret
= diff_tree_sha1(hash1
,hash2
,"",&p_Rev
->diffopt
);
564 diffcore_std(&p_Rev
->diffopt
);
565 memset(&p_Rev
->diffstat
, 0, sizeof(struct diffstat_t
));
566 for (i
= 0; i
< q
->nr
; i
++) {
567 struct diff_filepair
*p
= q
->queue
[i
];
568 //if (check_pair_status(p))
569 diff_flush_stat(p
, &p_Rev
->diffopt
, &p_Rev
->diffstat
);
580 int git_get_diff_file(GIT_DIFF diff
,GIT_FILE file
,int i
, char **newname
, char ** oldname
, int *status
, int *IsBin
, int *inc
, int *dec
)
582 struct diff_queue_struct
*q
= &diff_queued_diff
;
583 struct rev_info
*p_Rev
;
584 p_Rev
= (struct rev_info
*)diff
;
586 q
= (struct diff_queue_struct
*)file
;
593 *newname
= q
->queue
[i
]->two
->path
;
596 *oldname
= q
->queue
[i
]->one
->path
;
599 *status
= q
->queue
[i
]->status
;
601 if(p_Rev
->diffstat
.files
)
604 for(j
=0;j
<p_Rev
->diffstat
.nr
;j
++)
606 if(strcmp(*newname
,p_Rev
->diffstat
.files
[j
]->name
)==0)
609 if( j
== p_Rev
->diffstat
.nr
)
617 *IsBin
= p_Rev
->diffstat
.files
[j
]->is_binary
;
619 *inc
= p_Rev
->diffstat
.files
[j
]->added
;
621 *dec
= p_Rev
->diffstat
.files
[j
]->deleted
;
632 int git_read_tree(GIT_HASH hash
,read_tree_fn_t fn
, void *context
)
636 reprepare_packed_git();
637 root
= parse_tree_indirect(hash
);
644 ret
= read_tree_recursive(root
,NULL
,NULL
,0,NULL
,fn
,context
);
649 int git_add_exclude(const char *string
, const char *base
,
650 int baselen
, struct exclude_list
*which
)
652 add_exclude(string
, base
, baselen
, which
);
656 int git_create_exclude_list(EXCLUDE_LIST
*which
)
658 *which
= malloc(sizeof(struct exclude_list
));
659 memset(*which
,0,sizeof(struct exclude_list
));
663 int git_free_exclude_list(EXCLUDE_LIST which
)
666 struct exclude_list
*p
= (struct exclude_list
*) which
;
668 for(i
=0; i
<p
->nr
;i
++)
670 free(p
->excludes
[i
]);
677 int git_check_excluded_1(const char *pathname
,
678 int pathlen
, const char *basename
, int *dtype
,
681 return excluded_from_list(pathname
, pathlen
, basename
,dtype
,el
);
684 int git_get_notes(GIT_HASH hash
, char **p_note
)
689 format_display_notes(hash
, &sb
, "utf-8", 0);
690 *p_note
= strbuf_detach(&sb
,&size
);
697 int (*fn
)(int, const char **, const char *);
701 #define RUN_SETUP (1<<0)
702 #define USE_PAGER (1<<1)
704 * require working tree to be present -- anything uses this needs
705 * RUN_SETUP for reading from the configuration file.
707 #define NEED_WORK_TREE (1<<2)
709 const char git_usage_string
[] =
710 "git [--version] [--exec-path[=GIT_EXEC_PATH]] [--html-path]\n"
711 " [-p|--paginate|--no-pager] [--no-replace-objects]\n"
712 " [--bare] [--git-dir=GIT_DIR] [--work-tree=GIT_WORK_TREE]\n"
713 " [-c name=value] [--help]\n"
716 const char git_more_info_string
[] =
717 "See 'git help COMMAND' for more information on a specific command.";
719 /* returns 0 for "no pager", 1 for "use pager", and -1 for "not specified" */
720 int check_pager_config(const char *cmd
)
722 UNREFERENCED_PARAMETER(cmd
);
727 int git_run_cmd(char *cmd
, char *arg
)
734 static struct cmd_struct commands
[] = {
735 { "add", cmd_add
, RUN_SETUP
| NEED_WORK_TREE
},
736 { "stage", cmd_add
, RUN_SETUP
| NEED_WORK_TREE
},
737 { "annotate", cmd_annotate
, RUN_SETUP
},
738 { "apply", cmd_apply
},
739 { "archive", cmd_archive
},
740 { "bisect--helper", cmd_bisect__helper
, RUN_SETUP
| NEED_WORK_TREE
},
741 { "blame", cmd_blame
, RUN_SETUP
},
742 { "branch", cmd_branch
, RUN_SETUP
},
743 { "bundle", cmd_bundle
},
744 { "cat-file", cmd_cat_file
, RUN_SETUP
},
745 { "checkout", cmd_checkout
, RUN_SETUP
| NEED_WORK_TREE
},
746 { "checkout-index", cmd_checkout_index
,
747 RUN_SETUP
| NEED_WORK_TREE
},
748 { "check-ref-format", cmd_check_ref_format
},
749 { "check-attr", cmd_check_attr
, RUN_SETUP
},
750 { "cherry", cmd_cherry
, RUN_SETUP
},
751 { "cherry-pick", cmd_cherry_pick
, RUN_SETUP
| NEED_WORK_TREE
},
752 { "clone", cmd_clone
},
753 { "clean", cmd_clean
, RUN_SETUP
| NEED_WORK_TREE
},
754 { "commit", cmd_commit
, RUN_SETUP
| NEED_WORK_TREE
},
755 { "commit-tree", cmd_commit_tree
, RUN_SETUP
},
756 { "config", cmd_config
},
757 { "count-objects", cmd_count_objects
, RUN_SETUP
},
758 { "describe", cmd_describe
, RUN_SETUP
},
759 { "diff", cmd_diff
},
760 { "diff-files", cmd_diff_files
, RUN_SETUP
| NEED_WORK_TREE
},
761 { "diff-index", cmd_diff_index
, RUN_SETUP
},
762 { "diff-tree", cmd_diff_tree
, RUN_SETUP
},
763 { "fast-export", cmd_fast_export
, RUN_SETUP
},
764 { "fetch", cmd_fetch
, RUN_SETUP
},
765 { "fetch-pack", cmd_fetch_pack
, RUN_SETUP
},
766 { "fmt-merge-msg", cmd_fmt_merge_msg
, RUN_SETUP
},
767 { "for-each-ref", cmd_for_each_ref
, RUN_SETUP
},
768 { "format-patch", cmd_format_patch
, RUN_SETUP
},
769 { "fsck", cmd_fsck
, RUN_SETUP
},
770 { "fsck-objects", cmd_fsck
, RUN_SETUP
},
771 { "gc", cmd_gc
, RUN_SETUP
},
772 { "get-tar-commit-id", cmd_get_tar_commit_id
},
773 { "grep", cmd_grep
},
774 { "hash-object", cmd_hash_object
},
775 { "help", cmd_help
},
776 { "index-pack", cmd_index_pack
},
777 { "init", cmd_init_db
},
778 { "init-db", cmd_init_db
},
779 { "log", cmd_log
, RUN_SETUP
},
780 { "ls-files", cmd_ls_files
, RUN_SETUP
},
781 { "ls-tree", cmd_ls_tree
, RUN_SETUP
},
782 { "ls-remote", cmd_ls_remote
},
783 { "mailinfo", cmd_mailinfo
},
784 { "mailsplit", cmd_mailsplit
},
785 { "merge", cmd_merge
, RUN_SETUP
| NEED_WORK_TREE
},
786 { "merge-base", cmd_merge_base
, RUN_SETUP
},
787 { "merge-file", cmd_merge_file
},
788 { "merge-index", cmd_merge_index
, RUN_SETUP
},
789 { "merge-ours", cmd_merge_ours
, RUN_SETUP
},
790 { "merge-recursive", cmd_merge_recursive
, RUN_SETUP
| NEED_WORK_TREE
},
791 { "merge-recursive-ours", cmd_merge_recursive
, RUN_SETUP
| NEED_WORK_TREE
},
792 { "merge-recursive-theirs", cmd_merge_recursive
, RUN_SETUP
| NEED_WORK_TREE
},
793 { "merge-subtree", cmd_merge_recursive
, RUN_SETUP
| NEED_WORK_TREE
},
794 { "merge-tree", cmd_merge_tree
, RUN_SETUP
},
795 { "mktag", cmd_mktag
, RUN_SETUP
},
796 { "mktree", cmd_mktree
, RUN_SETUP
},
797 { "mv", cmd_mv
, RUN_SETUP
| NEED_WORK_TREE
},
798 { "name-rev", cmd_name_rev
, RUN_SETUP
},
799 { "notes", cmd_notes
, RUN_SETUP
},
800 { "pack-objects", cmd_pack_objects
, RUN_SETUP
},
801 { "pack-redundant", cmd_pack_redundant
, RUN_SETUP
},
802 { "patch-id", cmd_patch_id
},
803 { "peek-remote", cmd_ls_remote
},
804 { "pickaxe", cmd_blame
, RUN_SETUP
},
805 { "prune", cmd_prune
, RUN_SETUP
},
806 { "prune-packed", cmd_prune_packed
, RUN_SETUP
},
807 { "push", cmd_push
, RUN_SETUP
},
808 { "read-tree", cmd_read_tree
, RUN_SETUP
},
809 { "receive-pack", cmd_receive_pack
},
810 { "reflog", cmd_reflog
, RUN_SETUP
},
811 { "remote", cmd_remote
, RUN_SETUP
},
812 { "replace", cmd_replace
, RUN_SETUP
},
813 { "repo-config", cmd_config
},
814 { "rerere", cmd_rerere
, RUN_SETUP
},
815 { "reset", cmd_reset
, RUN_SETUP
},
816 { "rev-list", cmd_rev_list
, RUN_SETUP
},
817 { "rev-parse", cmd_rev_parse
},
818 { "revert", cmd_revert
, RUN_SETUP
| NEED_WORK_TREE
},
819 { "rm", cmd_rm
, RUN_SETUP
},
820 { "send-pack", cmd_send_pack
, RUN_SETUP
},
821 { "shortlog", cmd_shortlog
, USE_PAGER
},
822 { "show-branch", cmd_show_branch
, RUN_SETUP
},
823 { "show", cmd_show
, RUN_SETUP
},
824 { "status", cmd_status
, RUN_SETUP
| NEED_WORK_TREE
},
825 { "stripspace", cmd_stripspace
},
826 { "symbolic-ref", cmd_symbolic_ref
, RUN_SETUP
},
827 { "tag", cmd_tag
, RUN_SETUP
},
828 { "tar-tree", cmd_tar_tree
},
829 { "unpack-file", cmd_unpack_file
, RUN_SETUP
},
830 { "unpack-objects", cmd_unpack_objects
, RUN_SETUP
},
831 { "update-index", cmd_update_index
, RUN_SETUP
},
832 { "update-ref", cmd_update_ref
, RUN_SETUP
},
833 { "update-server-info", cmd_update_server_info
, RUN_SETUP
},
834 { "upload-archive", cmd_upload_archive
},
836 { "verify-tag", cmd_verify_tag
, RUN_SETUP
},
837 { "version", cmd_version
},
838 { "whatchanged", cmd_whatchanged
, RUN_SETUP
},
839 { "write-tree", cmd_write_tree
, RUN_SETUP
},
840 { "verify-pack", cmd_verify_pack
},
841 { "show-ref", cmd_show_ref
, RUN_SETUP
},
842 { "pack-refs", cmd_pack_refs
, RUN_SETUP
},
847 for(i
=0;i
< sizeof(commands
) / sizeof(struct cmd_struct
);i
++)
849 if(strcmp(cmd
,commands
[i
].cmd
)==0)
853 argv
= strtoargv(arg
,&argc
);
855 ret
= commands
[i
].fn(argc
, argv
, NULL
);
871 int git_for_each_ref_in(const char * refname
, each_ref_fn fn
, void * data
)
874 invalidate_ref_cache(NULL
);
875 ret
= for_each_ref_in(refname
, fn
, data
);
880 const char *git_resolve_ref(const char *ref
, unsigned char *sha1
, int reading
, int *flag
)
882 invalidate_ref_cache(NULL
);
883 return resolve_ref_unsafe(ref
,sha1
,reading
, flag
);
885 int git_for_each_reflog_ent(const char *ref
, each_reflog_ent_fn fn
, void *cb_data
)
887 return for_each_reflog_ent(ref
,fn
,cb_data
);
890 int git_deref_tag(const unsigned char *tagsha1
, GIT_HASH refhash
)
892 struct object
*obj
= NULL
;
893 obj
= parse_object(tagsha1
);
897 if (obj
->type
== OBJ_TAG
)
899 obj
= deref_tag(obj
, "", 0);
903 memcpy(refhash
, obj
->sha1
, sizeof(GIT_HASH
));
910 static int update_some(const unsigned char *sha1
, const char *base
, int baselen
,
911 const char *pathname
, unsigned mode
, int stage
, void *context
)
913 struct cache_entry
*ce
;
914 UNREFERENCED_PARAMETER(stage
);
916 ce
= (struct cache_entry
*)context
;
919 return READ_TREE_RECURSIVE
;
921 hashcpy(ce
->sha1
, sha1
);
922 memcpy(ce
->name
, base
, baselen
);
923 memcpy(ce
->name
+ baselen
, pathname
, strlen(pathname
));
924 ce
->ce_flags
= create_ce_flags(strlen(pathname
)+baselen
, 0);
925 ce
->ce_mode
= create_ce_mode(mode
);
930 int git_checkout_file(const char *ref
, const char *path
, const char *outputpath
)
932 struct cache_entry
*ce
;
936 struct checkout state
;
937 struct pathspec pathspec
;
938 const char *match
[2];
939 ret
= get_sha1(ref
, sha1
);
943 reprepare_packed_git();
944 root
= parse_tree_indirect(sha1
);
952 ce
= xcalloc(1, cache_entry_size(strlen(path
)));
957 init_pathspec(&pathspec
, match
);
958 pathspec
.items
[0].use_wildcard
= 0;
959 ret
= read_tree_recursive(root
, "", 0, 0, &pathspec
, update_some
, ce
);
960 free_pathspec(&pathspec
);
968 memset(&state
, 0, sizeof(state
));
970 state
.refresh_cache
= 0;
972 ret
= write_entry(ce
, outputpath
, &state
, 0);
985 static int get_config(const char *key_
, const char *value_
, void *cb
)
987 struct config_buf
*buf
;
988 buf
=(struct config_buf
*)cb
;
989 if(strcmp(key_
, buf
->key
))
993 strncpy(buf
->buf
,value_
,buf
->size
);
1007 // wchar_t wrapper for git_etc_gitconfig()
1008 const wchar_t *wget_msysgit_etc(void)
1010 static const wchar_t *etc_gitconfig
= NULL
;
1011 wchar_t wpointer
[MAX_PATH
];
1014 return etc_gitconfig
;
1016 if (xutftowcs_path(wpointer
, git_etc_gitconfig()) < 0)
1019 etc_gitconfig
= _wcsdup(wpointer
);
1021 return etc_gitconfig
;
1024 int git_get_config(const char *key
, char *buffer
, int size
, char *git_path
)
1026 char *local
, *global
, *globalxdg
;
1027 const char *home
, *system
;
1028 struct config_buf buf
;
1029 UNREFERENCED_PARAMETER(git_path
);
1035 home
= get_windows_home_directory();
1038 global
= xstrdup(mkpath("%s/.gitconfig", home
));
1039 globalxdg
= xstrdup(mkpath("%s/.config/git/config", home
));
1047 system
= git_etc_gitconfig();
1049 local
= git_pathdup("config");
1052 git_config_from_file(get_config
, local
, &buf
);
1053 if (!buf
.seen
&& global
)
1054 git_config_from_file(get_config
, global
, &buf
);
1055 if (!buf
.seen
&& globalxdg
)
1056 git_config_from_file(get_config
, globalxdg
, &buf
);
1057 if (!buf
.seen
&& system
)
1058 git_config_from_file(get_config
, system
, &buf
);
1070 // taken from msysgit: compat/mingw.c
1071 const char *get_windows_home_directory(void)
1073 static const char *home_directory
= NULL
;
1074 struct strbuf buf
= STRBUF_INIT
;
1077 return home_directory
;
1079 home_directory
= getenv("HOME");
1080 if (home_directory
&& *home_directory
)
1081 return home_directory
;
1083 strbuf_addf(&buf
, "%s/%s", getenv("HOMEDRIVE"), getenv("HOMEPATH"));
1084 home_directory
= strbuf_detach(&buf
, NULL
);
1086 return home_directory
;
1089 // wchar_t wrapper for get_windows_home_directory()
1090 const wchar_t *wget_windows_home_directory(void)
1092 static const wchar_t *home_directory
= NULL
;
1093 wchar_t wpointer
[MAX_PATH
];
1096 return home_directory
;
1098 if (xutftowcs_path(wpointer
, get_windows_home_directory()) < 0)
1101 home_directory
= _wcsdup(wpointer
);
1103 return home_directory
;
1106 int get_set_config(const char *key
, char *value
, CONFIG_TYPE type
,char *git_path
)
1108 char * config_exclusive_filename
= NULL
;
1109 UNREFERENCED_PARAMETER(git_path
);
1113 config_exclusive_filename
= git_pathdup("config");
1116 case CONFIG_XDGGLOBAL
:
1118 const char *home
= get_windows_home_directory();
1121 if (type
== CONFIG_GLOBAL
)
1122 config_exclusive_filename
= xstrdup(mkpath("%s/.gitconfig", home
));
1124 config_exclusive_filename
= xstrdup(mkpath("%s/.config/git/config", home
));
1130 if(!config_exclusive_filename
)
1133 return git_config_set_multivar_in_file(config_exclusive_filename
, key
, value
, NULL
, 0);
1136 struct mailmap_info
{
1141 struct mailmap_entry
{
1142 /* name and email for the simple mail-only case */
1146 /* name and email for the complex mail and name matching case */
1147 struct string_list namemap
;
1150 int git_read_mailmap(GIT_MAILMAP
*mailmap
)
1152 struct string_list
*map
;
1159 if (!(map
= (struct string_list
*)calloc(1, sizeof(struct string_list
))))
1162 if ((result
= read_mailmap(map
, NULL
)))
1169 const char * git_get_mailmap_author(GIT_MAILMAP mailmap
, const char *email2
, void *payload
, const char *(*author2_cb
)(void *))
1171 struct string_list
*map
;
1177 map
= (struct string_list
*)mailmap
;
1179 while (imax
>= imin
)
1181 int i
= imin
+ ((imax
- imin
) / 2);
1182 struct string_list_item
*si
= (struct string_list_item
*)&map
->items
[i
];
1183 struct mailmap_entry
*me
= (struct mailmap_entry
*)si
->util
;
1184 int comp
= strcmp(si
->string
, email2
);
1190 const char *author2
= author2_cb(payload
);
1192 for (j
= 0; j
< me
->namemap
.nr
; ++j
)
1194 struct string_list_item
*sj
= (struct string_list_item
*)&me
->namemap
.items
[j
];
1195 struct mailmap_info
*mi
= (struct mailmap_info
*)sj
->util
;
1197 if (!strcmp(sj
->string
, author2
))
1213 void git_clear_mailmap(GIT_MAILMAP mailmap
)
1218 clear_mailmap((struct string_list
*)mailmap
);