| #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 |