|
|
|
#ifndef COMMIT_H
|
|
|
|
#define COMMIT_H
|
|
|
|
|
|
|
|
#include "object.h"
|
|
|
|
#include "tree.h"
|
|
|
|
#include "strbuf.h"
|
|
|
|
#include "decorate.h"
|
|
|
|
|
|
|
|
struct commit_list {
|
|
|
|
struct commit *item;
|
|
|
|
struct commit_list *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct commit {
|
|
|
|
struct object object;
|
|
|
|
void *util;
|
|
|
|
unsigned int indegree;
|
|
|
|
unsigned long date;
|
|
|
|
struct commit_list *parents;
|
|
|
|
struct tree *tree;
|
|
|
|
char *buffer;
|
|
|
|
};
|
|
|
|
|
[PATCH] Avoid wasting memory in git-rev-list
As pointed out on the list, git-rev-list can use a lot of memory.
One low-hanging fruit is to free the commit buffer for commits that we
parse. By default, parse_commit() will save away the buffer, since a lot
of cases do want it, and re-reading it continually would be unnecessary.
However, in many cases the buffer isn't actually necessary and saving it
just wastes memory.
We could just free the buffer ourselves, but especially in git-rev-list,
we actually end up using the helper functions that automatically add
parent commits to the commit lists, so we don't actually control the
commit parsing directly.
Instead, just make this behaviour of "parse_commit()" a global flag.
Maybe this is a bit tasteless, but it's very simple, and it makes a
noticable difference in memory usage.
Before the change:
[torvalds@g5 linux]$ /usr/bin/time git-rev-list v2.6.12..HEAD > /dev/null
0.26user 0.02system 0:00.28elapsed 99%CPU (0avgtext+0avgdata 0maxresident)k
0inputs+0outputs (0major+3714minor)pagefaults 0swaps
after the change:
[torvalds@g5 linux]$ /usr/bin/time git-rev-list v2.6.12..HEAD > /dev/null
0.26user 0.00system 0:00.27elapsed 100%CPU (0avgtext+0avgdata 0maxresident)k
0inputs+0outputs (0major+2433minor)pagefaults 0swaps
note how the minor faults have decreased from 3714 pages to 2433 pages.
That's all due to the fewer anonymous pages allocated to hold the comment
buffers and their metadata.
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Signed-off-by: Junio C Hamano <junkio@cox.net>
20 years ago
|
|
|
extern int save_commit_buffer;
|
|
|
|
extern const char *commit_type;
|
|
|
|
|
|
|
|
/* While we can decorate any object with a name, it's only used for commits.. */
|
|
|
|
extern struct decoration name_decoration;
|
|
|
|
struct name_decoration {
|
|
|
|
struct name_decoration *next;
|
|
|
|
char name[1];
|
|
|
|
};
|
|
|
|
|
|
|
|
struct commit *lookup_commit(const unsigned char *sha1);
|
|
|
|
struct commit *lookup_commit_reference(const unsigned char *sha1);
|
|
|
|
struct commit *lookup_commit_reference_gently(const unsigned char *sha1,
|
|
|
|
int quiet);
|
|
|
|
|
|
|
|
int parse_commit_buffer(struct commit *item, void *buffer, unsigned long size);
|
|
|
|
|
|
|
|
int parse_commit(struct commit *item);
|
|
|
|
|
|
|
|
struct commit_list * commit_list_insert(struct commit *item, struct commit_list **list_p);
|
|
|
|
struct commit_list * insert_by_date(struct commit *item, struct commit_list **list);
|
|
|
|
|
|
|
|
void free_commit_list(struct commit_list *list);
|
|
|
|
|
|
|
|
void sort_by_date(struct commit_list **list);
|
|
|
|
|
|
|
|
/* Commit formats */
|
|
|
|
enum cmit_fmt {
|
|
|
|
CMIT_FMT_RAW,
|
|
|
|
CMIT_FMT_MEDIUM,
|
|
|
|
CMIT_FMT_DEFAULT = CMIT_FMT_MEDIUM,
|
|
|
|
CMIT_FMT_SHORT,
|
|
|
|
CMIT_FMT_FULL,
|
|
|
|
CMIT_FMT_FULLER,
|
|
|
|
CMIT_FMT_ONELINE,
|
|
|
|
CMIT_FMT_EMAIL,
|
|
|
|
CMIT_FMT_USERFORMAT,
|
|
|
|
|
|
|
|
CMIT_FMT_UNSPECIFIED,
|
|
|
|
};
|
|
|
|
|
|
|
|
extern int non_ascii(int);
|
|
|
|
extern enum cmit_fmt get_commit_format(const char *arg);
|
|
|
|
extern void format_commit_message(const struct commit *commit,
|
|
|
|
const void *format, struct strbuf *sb);
|
|
|
|
extern void pretty_print_commit(enum cmit_fmt fmt, const struct commit*,
|
|
|
|
struct strbuf *,
|
|
|
|
int abbrev, const char *subject,
|
|
|
|
const char *after_subject, enum date_mode,
|
|
|
|
int non_ascii_present);
|
|
|
|
|
|
|
|
/** Removes the first commit from a list sorted by date, and adds all
|
|
|
|
* of its parents.
|
|
|
|
**/
|
|
|
|
struct commit *pop_most_recent_commit(struct commit_list **list,
|
|
|
|
unsigned int mark);
|
|
|
|
|
|
|
|
struct commit *pop_commit(struct commit_list **stack);
|
|
|
|
|
|
|
|
void clear_commit_marks(struct commit *commit, unsigned int mark);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Performs an in-place topological sort of list supplied.
|
|
|
|
*
|
|
|
|
* invariant of resulting list is:
|
|
|
|
* a reachable from b => ord(b) < ord(a)
|
|
|
|
* in addition, when lifo == 0, commits on parallel tracks are
|
|
|
|
* sorted in the dates order.
|
|
|
|
*/
|
|
|
|
void sort_in_topological_order(struct commit_list ** list, int lifo);
|
|
|
|
|
|
|
|
struct commit_graft {
|
|
|
|
unsigned char sha1[20];
|
|
|
|
int nr_parent; /* < 0 if shallow commit */
|
|
|
|
unsigned char parent[FLEX_ARRAY][20]; /* more */
|
|
|
|
};
|
|
|
|
|
|
|
|
struct commit_graft *read_graft_line(char *buf, int len);
|
|
|
|
int register_commit_graft(struct commit_graft *, int);
|
|
|
|
int read_graft_file(const char *graft_file);
|
|
|
|
|
|
|
|
extern struct commit_list *get_merge_bases(struct commit *rev1, struct commit *rev2, int cleanup);
|
|
|
|
|
|
|
|
extern int register_shallow(const unsigned char *sha1);
|
|
|
|
extern int unregister_shallow(const unsigned char *sha1);
|
|
|
|
extern int write_shallow_commits(int fd, int use_pack_protocol);
|
|
|
|
extern int is_repository_shallow(void);
|
|
|
|
extern struct commit_list *get_shallow_commits(struct object_array *heads,
|
|
|
|
int depth, int shallow_flag, int not_shallow_flag);
|
|
|
|
|
|
|
|
int in_merge_bases(struct commit *, struct commit **, int);
|
|
|
|
|
|
|
|
extern int interactive_add(int argc, const char **argv, const char *prefix);
|
|
|
|
extern int rerere(void);
|
|
|
|
|
|
|
|
static inline int single_parent(struct commit *commit)
|
|
|
|
{
|
|
|
|
return commit->parents && !commit->parents->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* COMMIT_H */
|