Blame view

lib/ts_kmp.c 4.12 KB
2874c5fd2   Thomas Gleixner   treewide: Replace...
1
  // SPDX-License-Identifier: GPL-2.0-or-later
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
2
3
4
  /*
   * lib/ts_kmp.c		Knuth-Morris-Pratt text search implementation
   *
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
   * Authors:	Thomas Graf <tgraf@suug.ch>
   *
   * ==========================================================================
   * 
   *   Implements a linear-time string-matching algorithm due to Knuth,
   *   Morris, and Pratt [1]. Their algorithm avoids the explicit
   *   computation of the transition function DELTA altogether. Its
   *   matching time is O(n), for n being length(text), using just an
   *   auxiliary function PI[1..m], for m being length(pattern),
   *   precomputed from the pattern in time O(m). The array PI allows
   *   the transition function DELTA to be computed efficiently
   *   "on the fly" as needed. Roughly speaking, for any state
   *   "q" = 0,1,...,m and any character "a" in SIGMA, the value
   *   PI["q"] contains the information that is independent of "a" and
   *   is needed to compute DELTA("q", "a") [2]. Since the array PI
   *   has only m entries, whereas DELTA has O(m|SIGMA|) entries, we
   *   save a factor of |SIGMA| in the preprocessing time by computing
   *   PI rather than DELTA.
   *
   *   [1] Cormen, Leiserson, Rivest, Stein
   *       Introdcution to Algorithms, 2nd Edition, MIT Press
7433a8d6f   Randy Dunlap   textsearch: fix t...
26
   *   [2] See finite automaton theory
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
27
   */
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
28
29
30
  #include <linux/module.h>
  #include <linux/types.h>
  #include <linux/string.h>
2523c3fc2   Joonwoo Park   textsearch: ts_km...
31
  #include <linux/ctype.h>
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
32
33
34
35
36
37
  #include <linux/textsearch.h>
  
  struct ts_kmp
  {
  	u8 *		pattern;
  	unsigned int	pattern_len;
51022f871   Gustavo A. R. Silva   lib/ts_kmp.c: rep...
38
  	unsigned int	prefix_tbl[];
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
39
40
41
42
43
44
45
  };
  
  static unsigned int kmp_find(struct ts_config *conf, struct ts_state *state)
  {
  	struct ts_kmp *kmp = ts_config_priv(conf);
  	unsigned int i, q = 0, text_len, consumed = state->offset;
  	const u8 *text;
2523c3fc2   Joonwoo Park   textsearch: ts_km...
46
  	const int icase = conf->flags & TS_IGNORECASE;
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
47
48
49
50
51
52
53
54
  
  	for (;;) {
  		text_len = conf->get_next_block(consumed, &text, conf, state);
  
  		if (unlikely(text_len == 0))
  			break;
  
  		for (i = 0; i < text_len; i++) {
2523c3fc2   Joonwoo Park   textsearch: ts_km...
55
56
  			while (q > 0 && kmp->pattern[q]
  			    != (icase ? toupper(text[i]) : text[i]))
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
57
  				q = kmp->prefix_tbl[q - 1];
2523c3fc2   Joonwoo Park   textsearch: ts_km...
58
59
  			if (kmp->pattern[q]
  			    == (icase ? toupper(text[i]) : text[i]))
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
60
61
62
63
64
65
66
67
68
69
70
71
72
73
  				q++;
  			if (unlikely(q == kmp->pattern_len)) {
  				state->offset = consumed + i + 1;
  				return state->offset - kmp->pattern_len;
  			}
  		}
  
  		consumed += text_len;
  	}
  
  	return UINT_MAX;
  }
  
  static inline void compute_prefix_tbl(const u8 *pattern, unsigned int len,
2523c3fc2   Joonwoo Park   textsearch: ts_km...
74
  				      unsigned int *prefix_tbl, int flags)
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
75
76
  {
  	unsigned int k, q;
2523c3fc2   Joonwoo Park   textsearch: ts_km...
77
  	const u8 icase = flags & TS_IGNORECASE;
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
78
79
  
  	for (k = 0, q = 1; q < len; q++) {
2523c3fc2   Joonwoo Park   textsearch: ts_km...
80
81
  		while (k > 0 && (icase ? toupper(pattern[k]) : pattern[k])
  		    != (icase ? toupper(pattern[q]) : pattern[q]))
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
82
  			k = prefix_tbl[k-1];
2523c3fc2   Joonwoo Park   textsearch: ts_km...
83
84
  		if ((icase ? toupper(pattern[k]) : pattern[k])
  		    == (icase ? toupper(pattern[q]) : pattern[q]))
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
85
86
87
88
89
90
  			k++;
  		prefix_tbl[q] = k;
  	}
  }
  
  static struct ts_config *kmp_init(const void *pattern, unsigned int len,
2523c3fc2   Joonwoo Park   textsearch: ts_km...
91
  				  gfp_t gfp_mask, int flags)
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
92
93
94
  {
  	struct ts_config *conf;
  	struct ts_kmp *kmp;
2523c3fc2   Joonwoo Park   textsearch: ts_km...
95
  	int i;
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
96
97
98
99
100
101
  	unsigned int prefix_tbl_len = len * sizeof(unsigned int);
  	size_t priv_size = sizeof(*kmp) + len + prefix_tbl_len;
  
  	conf = alloc_ts_config(priv_size, gfp_mask);
  	if (IS_ERR(conf))
  		return conf;
2523c3fc2   Joonwoo Park   textsearch: ts_km...
102
  	conf->flags = flags;
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
103
104
  	kmp = ts_config_priv(conf);
  	kmp->pattern_len = len;
2523c3fc2   Joonwoo Park   textsearch: ts_km...
105
  	compute_prefix_tbl(pattern, len, kmp->prefix_tbl, flags);
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
106
  	kmp->pattern = (u8 *) kmp->prefix_tbl + prefix_tbl_len;
2523c3fc2   Joonwoo Park   textsearch: ts_km...
107
108
109
110
111
  	if (flags & TS_IGNORECASE)
  		for (i = 0; i < len; i++)
  			kmp->pattern[i] = toupper(((u8 *)pattern)[i]);
  	else
  		memcpy(kmp->pattern, pattern, len);
df3fb93ad   Thomas Graf   [LIB]: Knuth-Morr...
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
  
  	return conf;
  }
  
  static void *kmp_get_pattern(struct ts_config *conf)
  {
  	struct ts_kmp *kmp = ts_config_priv(conf);
  	return kmp->pattern;
  }
  
  static unsigned int kmp_get_pattern_len(struct ts_config *conf)
  {
  	struct ts_kmp *kmp = ts_config_priv(conf);
  	return kmp->pattern_len;
  }
  
  static struct ts_ops kmp_ops = {
  	.name		  = "kmp",
  	.find		  = kmp_find,
  	.init		  = kmp_init,
  	.get_pattern	  = kmp_get_pattern,
  	.get_pattern_len  = kmp_get_pattern_len,
  	.owner		  = THIS_MODULE,
  	.list		  = LIST_HEAD_INIT(kmp_ops.list)
  };
  
  static int __init init_kmp(void)
  {
  	return textsearch_register(&kmp_ops);
  }
  
  static void __exit exit_kmp(void)
  {
  	textsearch_unregister(&kmp_ops);
  }
  
  MODULE_LICENSE("GPL");
  
  module_init(init_kmp);
  module_exit(exit_kmp);