This patch adds a command line option analogous to that of GNU grep(1)'s -m / --max-count, which users might already be used to. This makes it possible to limit the amount of matches shown in the output while keeping the functionality of other options such as -C (show code context) or -p (show containing function), which would be difficult to do with a shell pipeline (e.g. head(1)). Signed-off-by: Carlos López 00xc@protonmail.com Signed-off-by: Junio C Hamano <gitster@pobox.com>
		
			
				
	
	
		
			263 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#ifndef GREP_H
 | 
						|
#define GREP_H
 | 
						|
#include "color.h"
 | 
						|
#ifdef USE_LIBPCRE2
 | 
						|
#define PCRE2_CODE_UNIT_WIDTH 8
 | 
						|
#include <pcre2.h>
 | 
						|
#if (PCRE2_MAJOR >= 10 && PCRE2_MINOR >= 36) || PCRE2_MAJOR >= 11
 | 
						|
#define GIT_PCRE2_VERSION_10_36_OR_HIGHER
 | 
						|
#endif
 | 
						|
#if (PCRE2_MAJOR >= 10 && PCRE2_MINOR >= 34) || PCRE2_MAJOR >= 11
 | 
						|
#define GIT_PCRE2_VERSION_10_34_OR_HIGHER
 | 
						|
#endif
 | 
						|
#else
 | 
						|
typedef int pcre2_code;
 | 
						|
typedef int pcre2_match_data;
 | 
						|
typedef int pcre2_compile_context;
 | 
						|
typedef int pcre2_general_context;
 | 
						|
#endif
 | 
						|
#ifndef PCRE2_MATCH_INVALID_UTF
 | 
						|
/* PCRE2_MATCH_* dummy also with !USE_LIBPCRE2, for test-pcre2-config.c */
 | 
						|
#define PCRE2_MATCH_INVALID_UTF 0
 | 
						|
#endif
 | 
						|
#include "thread-utils.h"
 | 
						|
#include "userdiff.h"
 | 
						|
 | 
						|
struct repository;
 | 
						|
 | 
						|
enum grep_pat_token {
 | 
						|
	GREP_PATTERN,
 | 
						|
	GREP_PATTERN_HEAD,
 | 
						|
	GREP_PATTERN_BODY,
 | 
						|
	GREP_AND,
 | 
						|
	GREP_OPEN_PAREN,
 | 
						|
	GREP_CLOSE_PAREN,
 | 
						|
	GREP_NOT,
 | 
						|
	GREP_OR
 | 
						|
};
 | 
						|
 | 
						|
enum grep_context {
 | 
						|
	GREP_CONTEXT_HEAD,
 | 
						|
	GREP_CONTEXT_BODY
 | 
						|
};
 | 
						|
 | 
						|
enum grep_header_field {
 | 
						|
	GREP_HEADER_FIELD_MIN = 0,
 | 
						|
	GREP_HEADER_AUTHOR = GREP_HEADER_FIELD_MIN,
 | 
						|
	GREP_HEADER_COMMITTER,
 | 
						|
	GREP_HEADER_REFLOG,
 | 
						|
 | 
						|
	/* Must be at the end of the enum */
 | 
						|
	GREP_HEADER_FIELD_MAX
 | 
						|
};
 | 
						|
 | 
						|
enum grep_color {
 | 
						|
	GREP_COLOR_CONTEXT,
 | 
						|
	GREP_COLOR_FILENAME,
 | 
						|
	GREP_COLOR_FUNCTION,
 | 
						|
	GREP_COLOR_LINENO,
 | 
						|
	GREP_COLOR_COLUMNNO,
 | 
						|
	GREP_COLOR_MATCH_CONTEXT,
 | 
						|
	GREP_COLOR_MATCH_SELECTED,
 | 
						|
	GREP_COLOR_SELECTED,
 | 
						|
	GREP_COLOR_SEP,
 | 
						|
	NR_GREP_COLORS
 | 
						|
};
 | 
						|
 | 
						|
struct grep_pat {
 | 
						|
	struct grep_pat *next;
 | 
						|
	const char *origin;
 | 
						|
	int no;
 | 
						|
	enum grep_pat_token token;
 | 
						|
	char *pattern;
 | 
						|
	size_t patternlen;
 | 
						|
	enum grep_header_field field;
 | 
						|
	regex_t regexp;
 | 
						|
	pcre2_code *pcre2_pattern;
 | 
						|
	pcre2_match_data *pcre2_match_data;
 | 
						|
	pcre2_compile_context *pcre2_compile_context;
 | 
						|
	pcre2_general_context *pcre2_general_context;
 | 
						|
	const uint8_t *pcre2_tables;
 | 
						|
	uint32_t pcre2_jit_on;
 | 
						|
	unsigned fixed:1;
 | 
						|
	unsigned is_fixed:1;
 | 
						|
	unsigned ignore_case:1;
 | 
						|
	unsigned word_regexp:1;
 | 
						|
};
 | 
						|
 | 
						|
enum grep_expr_node {
 | 
						|
	GREP_NODE_ATOM,
 | 
						|
	GREP_NODE_NOT,
 | 
						|
	GREP_NODE_AND,
 | 
						|
	GREP_NODE_TRUE,
 | 
						|
	GREP_NODE_OR
 | 
						|
};
 | 
						|
 | 
						|
enum grep_pattern_type {
 | 
						|
	GREP_PATTERN_TYPE_UNSPECIFIED = 0,
 | 
						|
	GREP_PATTERN_TYPE_BRE,
 | 
						|
	GREP_PATTERN_TYPE_ERE,
 | 
						|
	GREP_PATTERN_TYPE_FIXED,
 | 
						|
	GREP_PATTERN_TYPE_PCRE
 | 
						|
};
 | 
						|
 | 
						|
struct grep_expr {
 | 
						|
	enum grep_expr_node node;
 | 
						|
	unsigned hit;
 | 
						|
	union {
 | 
						|
		struct grep_pat *atom;
 | 
						|
		struct grep_expr *unary;
 | 
						|
		struct {
 | 
						|
			struct grep_expr *left;
 | 
						|
			struct grep_expr *right;
 | 
						|
		} binary;
 | 
						|
	} u;
 | 
						|
};
 | 
						|
 | 
						|
struct grep_opt {
 | 
						|
	struct grep_pat *pattern_list;
 | 
						|
	struct grep_pat **pattern_tail;
 | 
						|
	struct grep_pat *header_list;
 | 
						|
	struct grep_pat **header_tail;
 | 
						|
	struct grep_expr *pattern_expression;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * NEEDSWORK: See if we can remove this field, because the repository
 | 
						|
	 * should probably be per-source. That is, grep.c functions using this
 | 
						|
	 * field should probably start using "repo" in "struct grep_source"
 | 
						|
	 * instead.
 | 
						|
	 *
 | 
						|
	 * This is potentially the cause of at least one bug - "git grep"
 | 
						|
	 * using the textconv attributes from the superproject on the
 | 
						|
	 * submodules. See the failing "git grep --textconv" tests in
 | 
						|
	 * t7814-grep-recurse-submodules.sh for more information.
 | 
						|
	 */
 | 
						|
	struct repository *repo;
 | 
						|
 | 
						|
	int linenum;
 | 
						|
	int columnnum;
 | 
						|
	int invert;
 | 
						|
	int ignore_case;
 | 
						|
	int status_only;
 | 
						|
	int name_only;
 | 
						|
	int unmatch_name_only;
 | 
						|
	int count;
 | 
						|
	int word_regexp;
 | 
						|
	int all_match;
 | 
						|
	int no_body_match;
 | 
						|
	int body_hit;
 | 
						|
#define GREP_BINARY_DEFAULT	0
 | 
						|
#define GREP_BINARY_NOMATCH	1
 | 
						|
#define GREP_BINARY_TEXT	2
 | 
						|
	int binary;
 | 
						|
	int allow_textconv;
 | 
						|
	int extended;
 | 
						|
	int use_reflog_filter;
 | 
						|
	int relative;
 | 
						|
	int pathname;
 | 
						|
	int null_following_name;
 | 
						|
	int only_matching;
 | 
						|
	int color;
 | 
						|
	int max_depth;
 | 
						|
	int funcname;
 | 
						|
	int funcbody;
 | 
						|
	int extended_regexp_option;
 | 
						|
	enum grep_pattern_type pattern_type_option;
 | 
						|
	int ignore_locale;
 | 
						|
	char colors[NR_GREP_COLORS][COLOR_MAXLEN];
 | 
						|
	unsigned pre_context;
 | 
						|
	unsigned post_context;
 | 
						|
	unsigned last_shown;
 | 
						|
	int show_hunk_mark;
 | 
						|
	int file_break;
 | 
						|
	int heading;
 | 
						|
	int max_count;
 | 
						|
	void *priv;
 | 
						|
 | 
						|
	void (*output)(struct grep_opt *opt, const void *data, size_t size);
 | 
						|
	void *output_priv;
 | 
						|
};
 | 
						|
 | 
						|
#define GREP_OPT_INIT { \
 | 
						|
	.relative = 1, \
 | 
						|
	.pathname = 1, \
 | 
						|
	.max_depth = -1, \
 | 
						|
	.max_count = -1, \
 | 
						|
	.pattern_type_option = GREP_PATTERN_TYPE_UNSPECIFIED, \
 | 
						|
	.colors = { \
 | 
						|
		[GREP_COLOR_CONTEXT] = "", \
 | 
						|
		[GREP_COLOR_FILENAME] = GIT_COLOR_MAGENTA, \
 | 
						|
		[GREP_COLOR_FUNCTION] = "", \
 | 
						|
		[GREP_COLOR_LINENO] = GIT_COLOR_GREEN, \
 | 
						|
		[GREP_COLOR_COLUMNNO] = GIT_COLOR_GREEN, \
 | 
						|
		[GREP_COLOR_MATCH_CONTEXT] = GIT_COLOR_BOLD_RED, \
 | 
						|
		[GREP_COLOR_MATCH_SELECTED] = GIT_COLOR_BOLD_RED, \
 | 
						|
		[GREP_COLOR_SELECTED] = "", \
 | 
						|
		[GREP_COLOR_SEP] = GIT_COLOR_CYAN, \
 | 
						|
	}, \
 | 
						|
	.only_matching = 0, \
 | 
						|
	.color = -1, \
 | 
						|
	.output = std_output, \
 | 
						|
}
 | 
						|
 | 
						|
int grep_config(const char *var, const char *value, void *);
 | 
						|
void grep_init(struct grep_opt *, struct repository *repo);
 | 
						|
 | 
						|
void append_grep_pat(struct grep_opt *opt, const char *pat, size_t patlen, const char *origin, int no, enum grep_pat_token t);
 | 
						|
void append_grep_pattern(struct grep_opt *opt, const char *pat, const char *origin, int no, enum grep_pat_token t);
 | 
						|
void append_header_grep_pattern(struct grep_opt *, enum grep_header_field, const char *);
 | 
						|
void compile_grep_patterns(struct grep_opt *opt);
 | 
						|
void free_grep_patterns(struct grep_opt *opt);
 | 
						|
int grep_buffer(struct grep_opt *opt, const char *buf, unsigned long size);
 | 
						|
 | 
						|
/* The field parameter is only used to filter header patterns
 | 
						|
 * (where appropriate). If filtering isn't desirable
 | 
						|
 * GREP_HEADER_FIELD_MAX should be supplied.
 | 
						|
 */
 | 
						|
int grep_next_match(struct grep_opt *opt,
 | 
						|
		    const char *bol, const char *eol,
 | 
						|
		    enum grep_context ctx, regmatch_t *pmatch,
 | 
						|
		    enum grep_header_field field, int eflags);
 | 
						|
 | 
						|
struct grep_source {
 | 
						|
	char *name;
 | 
						|
 | 
						|
	enum grep_source_type {
 | 
						|
		GREP_SOURCE_OID,
 | 
						|
		GREP_SOURCE_FILE,
 | 
						|
		GREP_SOURCE_BUF,
 | 
						|
	} type;
 | 
						|
	void *identifier;
 | 
						|
	struct repository *repo; /* if GREP_SOURCE_OID */
 | 
						|
 | 
						|
	const char *buf;
 | 
						|
	unsigned long size;
 | 
						|
 | 
						|
	char *path; /* for attribute lookups */
 | 
						|
	struct userdiff_driver *driver;
 | 
						|
};
 | 
						|
 | 
						|
void grep_source_init_file(struct grep_source *gs, const char *name,
 | 
						|
			   const char *path);
 | 
						|
void grep_source_init_oid(struct grep_source *gs, const char *name,
 | 
						|
			  const char *path, const struct object_id *oid,
 | 
						|
			  struct repository *repo);
 | 
						|
void grep_source_clear_data(struct grep_source *gs);
 | 
						|
void grep_source_clear(struct grep_source *gs);
 | 
						|
void grep_source_load_driver(struct grep_source *gs,
 | 
						|
			     struct index_state *istate);
 | 
						|
 | 
						|
 | 
						|
int grep_source(struct grep_opt *opt, struct grep_source *gs);
 | 
						|
 | 
						|
struct grep_opt *grep_opt_dup(const struct grep_opt *opt);
 | 
						|
 | 
						|
/*
 | 
						|
 * Mutex used around access to the attributes machinery if
 | 
						|
 * opt->use_threads.  Must be initialized/destroyed by callers!
 | 
						|
 */
 | 
						|
extern int grep_use_locks;
 | 
						|
extern pthread_mutex_t grep_attr_mutex;
 | 
						|
 | 
						|
#endif
 |