xref: /haiku/headers/private/file_systems/QueryParser.h (revision 4c8e85b316c35a9161f5a1c50ad70bc91c83a76f)
1 /*
2  * Copyright 2001-2014, Axel Dörfler, axeld@pinc-software.de.
3  * Copyright 2010, Clemens Zeidler <haiku@clemens-zeidler.de>
4  * Copyright 2011, Ingo Weinhold, ingo_weinhold@gmx.de.
5  * This file may be used under the terms of the MIT License.
6  */
7 #ifndef _FILE_SYSTEMS_QUERY_PARSER_H
8 #define _FILE_SYSTEMS_QUERY_PARSER_H
9 
10 
11 /*!	Query parsing and evaluation
12 
13 	The pattern matching is roughly based on code originally written
14 	by J. Kercheval, and on code written by Kenneth Almquist, though
15 	it shares no code.
16 */
17 
18 
19 // The parser has a very static design, but it will do what is required.
20 //
21 // ParseOr(), ParseAnd(), ParseEquation() are guarantying the operator
22 // precedence, that is =,!=,>,<,>=,<= .. && .. ||.
23 // Apparently, the "!" (not) can only be used with brackets.
24 //
25 // If you think that there are too few NULL pointer checks in some places
26 // of the code, just read the beginning of the query constructor.
27 // The API is not fully available, just the Query and the Expression class
28 // are.
29 
30 #ifdef FS_SHELL
31 #	include <new>
32 
33 #	include "fssh_api_wrapper.h"
34 #	include "fssh_auto_deleter.h"
35 #else
36 #	include <dirent.h>
37 #	include <stdlib.h>
38 #	include <string.h>
39 
40 #	include <algorithm>
41 #	include <new>
42 
43 #	include <fs_interface.h>
44 #	include <fs_query.h>
45 #	include <TypeConstants.h>
46 
47 #	include <util/SinglyLinkedList.h>
48 #	include <util/Stack.h>
49 
50 #	include <query_private.h>
51 
52 #	include <lock.h>
53 #endif	// !FS_SHELL
54 
55 #include <file_systems/QueryParserUtils.h>
56 
57 
58 //#define DEBUG_QUERY
59 
60 #ifndef QUERY_RETURN_ERROR
61 #	define QUERY_RETURN_ERROR(error)	return (error)
62 #endif
63 
64 #ifndef QUERY_REPORT_ERROR
65 #	define QUERY_REPORT_ERROR(error)	do {} while (false)
66 #endif
67 
68 #ifndef QUERY_FATAL
69 #	define QUERY_FATAL(message...)		panic(message)
70 #endif
71 
72 #ifndef QUERY_INFORM
73 #	define QUERY_INFORM(message...)		dprintf(message)
74 #endif
75 
76 #ifndef QUERY_D
77 #	define QUERY_D(block)
78 #endif
79 
80 
81 namespace QueryParser {
82 
83 
84 template<typename QueryPolicy> class Equation;
85 template<typename QueryPolicy> class Expression;
86 template<typename QueryPolicy> class Term;
87 template<typename QueryPolicy> class Query;
88 
89 
90 enum ops {
91 	OP_NONE,
92 
93 	OP_AND,
94 	OP_OR,
95 
96 	OP_EQUATION,
97 		// is only used for invalid equations
98 
99 	OP_EQUAL,
100 	OP_UNEQUAL,
101 	OP_GREATER_THAN,
102 	OP_LESS_THAN,
103 	OP_GREATER_THAN_OR_EQUAL,
104 	OP_LESS_THAN_OR_EQUAL,
105 };
106 
107 
108 template<typename QueryPolicy>
109 union value {
110 	int64	Int64;
111 	uint64	Uint64;
112 	int32	Int32;
113 	uint32	Uint32;
114 	float	Float;
115 	double	Double;
116 	char	String[QueryPolicy::kMaxFileNameLength];
117 };
118 
119 
120 template<typename QueryPolicy>
121 class Query {
122 public:
123 			typedef typename QueryPolicy::Entry Entry;
124 			typedef typename QueryPolicy::Index Index;
125 			typedef typename QueryPolicy::IndexIterator IndexIterator;
126 			typedef typename QueryPolicy::Node Node;
127 			typedef typename QueryPolicy::Context Context;
128 
129 public:
130 							Query(Context* context,
131 								Expression<QueryPolicy>* expression,
132 								uint32 flags, port_id port, uint32 token);
133 							~Query();
134 
135 	static	status_t		Create(Context* context, const char* queryString,
136 								uint32 flags, port_id port, uint32 token,
137 								Query<QueryPolicy>*& _query);
138 
139 			status_t		Rewind();
140 	inline	status_t		GetNextEntry(struct dirent* dirent, size_t size);
141 
142 			void			LiveUpdate(Entry* entry, Node* node,
143 								const char* attribute, int32 type,
144 								const uint8* oldKey, size_t oldLength,
145 								const uint8* newKey, size_t newLength);
146 			void			LiveUpdateRenameMove(Entry* entry, Node* node,
147 								ino_t oldDirectoryID, const char* oldName,
148 								size_t oldLength, ino_t newDirectoryID,
149 								const char* newName, size_t newLength);
150 
151 			Expression<QueryPolicy>* GetExpression() const
152 								{ return fExpression; }
153 
154 			uint32			Flags() const
155 								{ return fFlags; }
156 
157 private:
158 			status_t		_GetNextEntry(struct dirent* dirent, size_t size);
159 			void			_SendEntryNotification(Entry* entry,
160 								status_t (*notify)(port_id, int32, dev_t, ino_t,
161 									const char*, ino_t));
162 
163 private:
164 			Context*		fContext;
165 			Expression<QueryPolicy>* fExpression;
166 			Equation<QueryPolicy>* fCurrent;
167 			IndexIterator*	fIterator;
168 			Index			fIndex;
169 			Stack<Equation<QueryPolicy>*> fStack;
170 
171 			uint32			fFlags;
172 			port_id			fPort;
173 			int32			fToken;
174 			bool			fNeedsEntry;
175 };
176 
177 
178 /*!	Abstract base class for the operator/equation classes.
179 */
180 template<typename QueryPolicy>
181 class Term {
182 public:
183 			typedef typename QueryPolicy::Entry Entry;
184 			typedef typename QueryPolicy::Index Index;
185 			typedef typename QueryPolicy::IndexIterator IndexIterator;
186 			typedef typename QueryPolicy::Node Node;
187 			typedef typename QueryPolicy::Context Context;
188 
189 public:
190 						Term(int8 op) : fOp(op), fParent(NULL) {}
191 	virtual				~Term() {}
192 
193 			int8		Op() const { return fOp; }
194 
195 			void		SetParent(Term<QueryPolicy>* parent)
196 							{ fParent = parent; }
197 			Term<QueryPolicy>* Parent() const { return fParent; }
198 
199 	virtual	status_t	Match(Entry* entry, Node* node,
200 							const char* attribute = NULL, int32 type = 0,
201 							const uint8* key = NULL, size_t size = 0) = 0;
202 	virtual	void		Complement() = 0;
203 
204 	virtual	void		CalculateScore(Index& index) = 0;
205 	virtual	int32		Score() const = 0;
206 
207 	virtual	status_t	InitCheck() = 0;
208 
209 	virtual	bool		NeedsEntry() = 0;
210 
211 #ifdef DEBUG_QUERY
212 	virtual	void		PrintToStream() = 0;
213 #endif
214 
215 protected:
216 			int8		fOp;
217 			Term<QueryPolicy>* fParent;
218 };
219 
220 
221 /*!	An Equation object represents an "attribute-equation operator-value" pair.
222 
223 	Although an Equation object is quite independent from the volume on which
224 	the query is run, there are some dependencies that are produced while
225 	querying:
226 	The type/size of the value, the score, and if it has an index or not.
227 	So you could run more than one query on the same volume, but it might return
228 	wrong values when it runs concurrently on another volume.
229 	That's not an issue right now, because we run single-threaded and don't use
230 	queries more than once.
231 */
232 template<typename QueryPolicy>
233 class Equation : public Term<QueryPolicy> {
234 public:
235 			typedef typename QueryPolicy::Entry Entry;
236 			typedef typename QueryPolicy::Index Index;
237 			typedef typename QueryPolicy::IndexIterator IndexIterator;
238 			typedef typename QueryPolicy::Node Node;
239 			typedef typename QueryPolicy::Context Context;
240 
241 public:
242 						Equation(char** expression);
243 	virtual				~Equation();
244 
245 	virtual	status_t	InitCheck();
246 
247 			status_t	ParseQuotedString(char** _start, char** _end);
248 			char*		CopyString(char* start, char* end);
249 
250 	virtual	status_t	Match(Entry* entry, Node* node,
251 							const char* attribute = NULL, int32 type = 0,
252 							const uint8* key = NULL, size_t size = 0);
253 	virtual void		Complement();
254 
255 			status_t	PrepareQuery(Context* context, Index& index,
256 							IndexIterator** iterator, bool queryNonIndexed);
257 			status_t	GetNextMatching(Context* context,
258 							IndexIterator* iterator, struct dirent* dirent,
259 							size_t bufferSize);
260 
261 	virtual	void		CalculateScore(Index &index);
262 	virtual	int32		Score() const { return fScore; }
263 
264 	virtual	bool		NeedsEntry();
265 
266 #ifdef DEBUG_QUERY
267 	virtual	void		PrintToStream();
268 #endif
269 
270 private:
271 						Equation(const Equation& other);
272 						Equation& operator=(const Equation& other);
273 							// no implementation
274 
275 			status_t	ConvertValue(type_code type);
276 			bool		CompareTo(const uint8* value, size_t size);
277 			uint8*		Value() const { return (uint8*)&fValue; }
278 			status_t	MatchEmptyString();
279 
280 			char*		fAttribute;
281 			char*		fString;
282 			union value<QueryPolicy> fValue;
283 			type_code	fType;
284 			size_t		fSize;
285 			bool		fIsPattern;
286 
287 			int32		fScore;
288 			bool		fHasIndex;
289 };
290 
291 
292 /*!	The Operator class does not represent a generic operator, but only those
293 	that combine two equations, namely "or", and "and".
294 */
295 template<typename QueryPolicy>
296 class Operator : public Term<QueryPolicy> {
297 public:
298 			typedef typename QueryPolicy::Entry Entry;
299 			typedef typename QueryPolicy::Index Index;
300 			typedef typename QueryPolicy::IndexIterator IndexIterator;
301 			typedef typename QueryPolicy::Node Node;
302 			typedef typename QueryPolicy::Context Context;
303 
304 public:
305 						Operator(Term<QueryPolicy>* left, int8 op,
306 							Term<QueryPolicy>* right);
307 	virtual				~Operator();
308 
309 			Term<QueryPolicy>* Left() const { return fLeft; }
310 			Term<QueryPolicy>* Right() const { return fRight; }
311 
312 	virtual	status_t	Match(Entry* entry, Node* node,
313 							const char* attribute = NULL, int32 type = 0,
314 							const uint8* key = NULL, size_t size = 0);
315 	virtual	void		Complement();
316 
317 	virtual	void		CalculateScore(Index& index);
318 	virtual	int32		Score() const;
319 
320 	virtual	status_t	InitCheck();
321 
322 	virtual	bool		NeedsEntry();
323 
324 #ifdef DEBUG_QUERY
325 	virtual	void		PrintToStream();
326 #endif
327 
328 private:
329 						Operator(const Operator& other);
330 						Operator& operator=(const Operator& other);
331 							// no implementation
332 
333 			Term<QueryPolicy>* fLeft;
334 			Term<QueryPolicy>* fRight;
335 };
336 
337 
338 template<typename QueryPolicy>
339 class Expression {
340 public:
341 			typedef typename QueryPolicy::Entry Entry;
342 			typedef typename QueryPolicy::Index Index;
343 			typedef typename QueryPolicy::IndexIterator IndexIterator;
344 			typedef typename QueryPolicy::Node Node;
345 			typedef typename QueryPolicy::Context Context;
346 
347 public:
348 							Expression(char* expr);
349 							~Expression();
350 
351 			status_t		InitCheck();
352 			const char*		Position() const { return fPosition; }
353 			Term<QueryPolicy>* Root() const { return fTerm; }
354 
355 protected:
356 			Term<QueryPolicy>* ParseOr(char** expr);
357 			Term<QueryPolicy>* ParseAnd(char** expr);
358 			Term<QueryPolicy>* ParseEquation(char** expr);
359 
360 			bool			IsOperator(char** expr, char op);
361 
362 private:
363 							Expression(const Expression& other);
364 							Expression& operator=(const Expression& other);
365 								// no implementation
366 
367 			char*			fPosition;
368 			Term<QueryPolicy>* fTerm;
369 };
370 
371 
372 //	#pragma mark -
373 
374 
375 template<typename QueryPolicy>
376 Equation<QueryPolicy>::Equation(char** expr)
377 	:
378 	Term<QueryPolicy>(OP_EQUATION),
379 	fAttribute(NULL),
380 	fString(NULL),
381 	fType(0),
382 	fIsPattern(false)
383 {
384 	char* string = *expr;
385 	char* start = string;
386 	char* end = NULL;
387 
388 	// Since the equation is the integral part of any query, we're just parsing
389 	// the whole thing here.
390 	// The whitespace at the start is already removed in
391 	// Expression::ParseEquation()
392 
393 	if (*start == '"' || *start == '\'') {
394 		// string is quoted (start has to be on the beginning of a string)
395 		if (ParseQuotedString(&start, &end) < B_OK)
396 			return;
397 
398 		// set string to a valid start of the equation symbol
399 		string = end + 2;
400 		skipWhitespace(&string);
401 		if (*string != '=' && *string != '<' && *string != '>'
402 			&& *string != '!') {
403 			*expr = string;
404 			return;
405 		}
406 	} else {
407 		// search the (in)equation for the actual equation symbol (and for other operators
408 		// in case the equation is malformed)
409 		while (*string && *string != '=' && *string != '<' && *string != '>'
410 			&& *string != '!' && *string != '&' && *string != '|') {
411 			string++;
412 		}
413 
414 		// get the attribute string	(and trim whitespace), in case
415 		// the string was not quoted
416 		end = string - 1;
417 		skipWhitespaceReverse(&end, start);
418 	}
419 
420 	// attribute string is empty (which is not allowed)
421 	if (start > end)
422 		return;
423 
424 	// At this point, "start" points to the beginning of the string, "end"
425 	// points to the last character of the string, and "string" points to the
426 	// first character of the equation symbol
427 
428 	// test for the right symbol (as this doesn't need any memory)
429 	switch (*string) {
430 		case '=':
431 			Term<QueryPolicy>::fOp = OP_EQUAL;
432 			break;
433 		case '>':
434 			Term<QueryPolicy>::fOp = *(string + 1) == '='
435 				? OP_GREATER_THAN_OR_EQUAL : OP_GREATER_THAN;
436 			break;
437 		case '<':
438 			Term<QueryPolicy>::fOp = *(string + 1) == '='
439 				? OP_LESS_THAN_OR_EQUAL : OP_LESS_THAN;
440 			break;
441 		case '!':
442 			if (*(string + 1) != '=')
443 				return;
444 			Term<QueryPolicy>::fOp = OP_UNEQUAL;
445 			break;
446 
447 		// any invalid characters will be rejected
448 		default:
449 			*expr = string;
450 			return;
451 	}
452 
453 	// lets change "start" to point to the first character after the symbol
454 	if (*(string + 1) == '=')
455 		string++;
456 	string++;
457 	skipWhitespace(&string);
458 
459 	// allocate & copy the attribute string
460 
461 	fAttribute = CopyString(start, end);
462 	if (fAttribute == NULL)
463 		return;
464 
465 	start = string;
466 	if (*start == '"' || *start == '\'') {
467 		// string is quoted (start has to be on the beginning of a string)
468 		if (ParseQuotedString(&start, &end) < B_OK)
469 			return;
470 
471 		string = end + 2;
472 		skipWhitespace(&string);
473 	} else {
474 		while (*string && *string != '&' && *string != '|' && *string != ')')
475 			string++;
476 
477 		end = string - 1;
478 		skipWhitespaceReverse(&end, start);
479 	}
480 
481 	// at this point, "start" will point to the first character of the value,
482 	// "end" will point to its last character, and "start" to the first non-
483 	// whitespace character after the value string
484 
485 	fString = CopyString(start, end);
486 	if (fString == NULL)
487 		return;
488 
489 	// patterns are only allowed for these operations (and strings)
490 	if (Term<QueryPolicy>::fOp == OP_EQUAL
491 		|| Term<QueryPolicy>::fOp == OP_UNEQUAL) {
492 		fIsPattern = isPattern(fString);
493 		if (fIsPattern && isValidPattern(fString) < B_OK) {
494 			// we only want to have valid patterns; setting fString
495 			// to NULL will cause InitCheck() to fail
496 			free(fString);
497 			fString = NULL;
498 		}
499 	}
500 
501 	*expr = string;
502 }
503 
504 
505 template<typename QueryPolicy>
506 Equation<QueryPolicy>::~Equation()
507 {
508 	free(fAttribute);
509 	free(fString);
510 }
511 
512 
513 template<typename QueryPolicy>
514 status_t
515 Equation<QueryPolicy>::InitCheck()
516 {
517 	if (fAttribute == NULL || fString == NULL
518 		|| Term<QueryPolicy>::fOp == OP_NONE) {
519 		return B_BAD_VALUE;
520 	}
521 
522 	return B_OK;
523 }
524 
525 
526 template<typename QueryPolicy>
527 status_t
528 Equation<QueryPolicy>::ParseQuotedString(char** _start, char** _end)
529 {
530 	char* start = *_start;
531 	char quote = *start++;
532 	char* end = start;
533 
534 	for (; *end && *end != quote; end++) {
535 		if (*end == '\\')
536 			end++;
537 	}
538 	if (*end == '\0')
539 		return B_BAD_VALUE;
540 
541 	*_start = start;
542 	*_end = end - 1;
543 
544 	return B_OK;
545 }
546 
547 
548 template<typename QueryPolicy>
549 char*
550 Equation<QueryPolicy>::CopyString(char* start, char* end)
551 {
552 	// end points to the last character of the string - and the length
553 	// also has to include the null-termination
554 	int32 length = end + 2 - start;
555 	// just to make sure; since that's the max. attribute name length and
556 	// the max. string in an index, it make sense to have it that way
557 	if (length > QueryPolicy::kMaxFileNameLength || length <= 0)
558 		return NULL;
559 
560 	char* copy = (char*)malloc(length);
561 	if (copy == NULL)
562 		return NULL;
563 
564 	memcpy(copy, start, length - 1);
565 	copy[length - 1] = '\0';
566 
567 	return copy;
568 }
569 
570 
571 template<typename QueryPolicy>
572 status_t
573 Equation<QueryPolicy>::ConvertValue(type_code type)
574 {
575 	// Has the type already been converted?
576 	if (type == fType)
577 		return B_OK;
578 
579 	char* string = fString;
580 
581 	switch (type) {
582 		case B_MIME_STRING_TYPE:
583 			type = B_STRING_TYPE;
584 			// supposed to fall through
585 		case B_STRING_TYPE:
586 			strncpy(fValue.String, string, QueryPolicy::kMaxFileNameLength);
587 			fValue.String[QueryPolicy::kMaxFileNameLength - 1] = '\0';
588 			fSize = strlen(fValue.String);
589 			break;
590 		case B_INT32_TYPE:
591 			fValue.Int32 = strtol(string, &string, 0);
592 			fSize = sizeof(int32);
593 			break;
594 		case B_UINT32_TYPE:
595 			fValue.Int32 = strtoul(string, &string, 0);
596 			fSize = sizeof(uint32);
597 			break;
598 		case B_INT64_TYPE:
599 			fValue.Int64 = strtoll(string, &string, 0);
600 			fSize = sizeof(int64);
601 			break;
602 		case B_UINT64_TYPE:
603 			fValue.Uint64 = strtoull(string, &string, 0);
604 			fSize = sizeof(uint64);
605 			break;
606 		case B_FLOAT_TYPE:
607 			fValue.Float = strtod(string, &string);
608 			fSize = sizeof(float);
609 			break;
610 		case B_DOUBLE_TYPE:
611 			fValue.Double = strtod(string, &string);
612 			fSize = sizeof(double);
613 			break;
614 		default:
615 			QUERY_FATAL("query value conversion to 0x%x requested!\n",
616 				(int)type);
617 			// should we fail here or just do a safety int32 conversion?
618 			return B_ERROR;
619 	}
620 
621 	fType = type;
622 
623 	// patterns are only allowed for string types
624 	if (fType != B_STRING_TYPE && fIsPattern)
625 		fIsPattern = false;
626 
627 	return B_OK;
628 }
629 
630 
631 /*!	Returns true when the key matches the equation. You have to
632 	call ConvertValue() before this one.
633 */
634 template<typename QueryPolicy>
635 bool
636 Equation<QueryPolicy>::CompareTo(const uint8* value, size_t size)
637 {
638 	int32 compare;
639 
640 	// fIsPattern is only true if it's a string type, and fOp OP_EQUAL, or
641 	// OP_UNEQUAL
642 	if (fIsPattern) {
643 		// we have already validated the pattern, so we don't check for failing
644 		// here - if something is broken, and matchString() returns an error,
645 		// we just don't match
646 		compare = matchString(fValue.String, (char*)value) == MATCH_OK ? 0 : 1;
647 	} else
648 		compare = compareKeys(fType, value, size, Value(), fSize);
649 
650 	switch (Term<QueryPolicy>::fOp) {
651 		case OP_EQUAL:
652 			return compare == 0;
653 		case OP_UNEQUAL:
654 			return compare != 0;
655 		case OP_LESS_THAN:
656 			return compare < 0;
657 		case OP_LESS_THAN_OR_EQUAL:
658 			return compare <= 0;
659 		case OP_GREATER_THAN:
660 			return compare > 0;
661 		case OP_GREATER_THAN_OR_EQUAL:
662 			return compare >= 0;
663 	}
664 	QUERY_FATAL("Unknown/Unsupported operation: %d\n", Term<QueryPolicy>::fOp);
665 	return false;
666 }
667 
668 
669 template<typename QueryPolicy>
670 void
671 Equation<QueryPolicy>::Complement()
672 {
673 	QUERY_D(if (fOp <= OP_EQUATION || fOp > OP_LESS_THAN_OR_EQUAL) {
674 		QUERY_FATAL("op out of range!\n");
675 		return;
676 	});
677 
678 	int8 complementOp[] = {OP_UNEQUAL, OP_EQUAL, OP_LESS_THAN_OR_EQUAL,
679 			OP_GREATER_THAN_OR_EQUAL, OP_LESS_THAN, OP_GREATER_THAN};
680 	Term<QueryPolicy>::fOp = complementOp[Term<QueryPolicy>::fOp - OP_EQUAL];
681 }
682 
683 
684 template<typename QueryPolicy>
685 status_t
686 Equation<QueryPolicy>::MatchEmptyString()
687 {
688 	// There is no matching attribute, we will just bail out if we
689 	// already know that our value is not of a string type.
690 	// If not, it will be converted to a string - and then be compared with "".
691 	// That's why we have to call ConvertValue() here - but it will be
692 	// a cheap call for the next time
693 	// TODO: Should we do this only for OP_UNEQUAL?
694 	if (fType != 0 && fType != B_STRING_TYPE)
695 		return NO_MATCH;
696 
697 	status_t status = ConvertValue(B_STRING_TYPE);
698 	if (status == B_OK)
699 		status = CompareTo((const uint8*)"", fSize) ? MATCH_OK : NO_MATCH;
700 
701 	return status;
702 }
703 
704 
705 /*!	Matches the node's attribute value with the equation.
706 	Returns MATCH_OK if it matches, NO_MATCH if not, < 0 if something went
707 	wrong.
708 */
709 template<typename QueryPolicy>
710 status_t
711 Equation<QueryPolicy>::Match(Entry* entry, Node* node,
712 	const char* attributeName, int32 type, const uint8* key, size_t size)
713 {
714 	// get a pointer to the attribute in question
715 	union value<QueryPolicy> value;
716 	uint8* buffer = (uint8*)&value;
717 	const size_t bufferSize = sizeof(value);
718 
719 	// first, check if we are matching for a live query and use that value
720 	if (attributeName != NULL && !strcmp(fAttribute, attributeName)) {
721 		if (key == NULL) {
722 			if (type == B_STRING_TYPE)
723 				return MatchEmptyString();
724 
725 			return NO_MATCH;
726 		}
727 		buffer = const_cast<uint8*>(key);
728 	} else if (!strcmp(fAttribute, "name")) {
729 		// if not, check for "fake" attributes ("name", "size", "last_modified")
730 		if (entry == NULL)
731 			return B_ERROR;
732 		buffer = (uint8*)QueryPolicy::EntryGetNameNoCopy(entry, buffer,
733 			sizeof(value));
734 		if (buffer == NULL)
735 			return B_ERROR;
736 
737 		type = B_STRING_TYPE;
738 		size = strlen((const char*)buffer);
739 	} else if (!strcmp(fAttribute, "size")) {
740 		value.Int64 = QueryPolicy::NodeGetSize(node);
741 		type = B_INT64_TYPE;
742 	} else if (!strcmp(fAttribute, "last_modified")) {
743 		value.Int32 = QueryPolicy::NodeGetLastModifiedTime(node);
744 		type = B_INT32_TYPE;
745 	} else {
746 		// then for attributes
747 		size = bufferSize;
748 		if (QueryPolicy::NodeGetAttribute(node, fAttribute, buffer, &size,
749 				&type) != B_OK) {
750 			return MatchEmptyString();
751 		}
752 	}
753 
754 	// prepare own value for use, if it is possible to convert it
755 	status_t status = ConvertValue(type);
756 	if (status == B_OK)
757 		status = CompareTo(buffer, size) ? MATCH_OK : NO_MATCH;
758 
759 	QUERY_RETURN_ERROR(status);
760 }
761 
762 
763 template<typename QueryPolicy>
764 void
765 Equation<QueryPolicy>::CalculateScore(Index &index)
766 {
767 	// As always, these values could be tuned and refined.
768 	// And the code could also need some real world testing :-)
769 
770 	// do we have to operate on a "foreign" index?
771 	if (Term<QueryPolicy>::fOp == OP_UNEQUAL
772 		|| QueryPolicy::IndexSetTo(index, fAttribute) < B_OK) {
773 		fScore = 0;
774 		return;
775 	}
776 
777 	// if we have a pattern, how much does it help our search?
778 	if (fIsPattern)
779 		fScore = getFirstPatternSymbol(fString) << 3;
780 	else {
781 		// Score by operator
782 		if (Term<QueryPolicy>::fOp == OP_EQUAL) {
783 			// higher than pattern="255 chars+*"
784 			fScore = 2048;
785 		} else {
786 			// the pattern search is regarded cheaper when you have at
787 			// least one character to set your index to
788 			fScore = 5;
789 		}
790 	}
791 
792 	// take index size into account
793 	fScore = QueryPolicy::IndexGetWeightedScore(index, fScore);
794 }
795 
796 
797 template<typename QueryPolicy>
798 status_t
799 Equation<QueryPolicy>::PrepareQuery(Context* /*context*/, Index& index,
800 	IndexIterator** iterator, bool queryNonIndexed)
801 {
802 	status_t status = QueryPolicy::IndexSetTo(index, fAttribute);
803 
804 	// if we should query attributes without an index, we can just proceed here
805 	if (status != B_OK && !queryNonIndexed)
806 		return B_ENTRY_NOT_FOUND;
807 
808 	type_code type;
809 
810 	// Special case for OP_UNEQUAL - it will always operate through the whole
811 	// index but we need the call to the original index to get the correct type
812 	if (status != B_OK || Term<QueryPolicy>::fOp == OP_UNEQUAL) {
813 		// Try to get an index that holds all files (name)
814 		// Also sets the default type for all attributes without index
815 		// to string.
816 		type = status < B_OK ? B_STRING_TYPE : QueryPolicy::IndexGetType(index);
817 
818 		if (QueryPolicy::IndexSetTo(index, "name") != B_OK)
819 			return B_ENTRY_NOT_FOUND;
820 
821 		fHasIndex = false;
822 	} else {
823 		fHasIndex = true;
824 		type = QueryPolicy::IndexGetType(index);
825 	}
826 
827 	if (ConvertValue(type) < B_OK)
828 		return B_BAD_VALUE;
829 
830 	*iterator = QueryPolicy::IndexCreateIterator(index);
831 	if (*iterator == NULL)
832 		return B_NO_MEMORY;
833 
834 	if ((Term<QueryPolicy>::fOp == OP_EQUAL
835 			|| Term<QueryPolicy>::fOp == OP_GREATER_THAN
836 			|| Term<QueryPolicy>::fOp == OP_GREATER_THAN_OR_EQUAL || fIsPattern)
837 		&& fHasIndex) {
838 		// set iterator to the exact position
839 
840 		int32 keySize = QueryPolicy::IndexGetKeySize(index);
841 
842 		// At this point, fIsPattern is only true if it's a string type, and fOp
843 		// is either OP_EQUAL or OP_UNEQUAL
844 		if (fIsPattern) {
845 			// let's see if we can use the beginning of the key for positioning
846 			// the iterator and adjust the key size; if not, just leave the
847 			// iterator at the start and return success
848 			keySize = getFirstPatternSymbol(fString);
849 			if (keySize <= 0)
850 				return B_OK;
851 		}
852 
853 		if (keySize == 0) {
854 			// B_STRING_TYPE doesn't have a fixed length, so it was set
855 			// to 0 before - we compute the correct value here
856 			if (fType == B_STRING_TYPE) {
857 				keySize = strlen(fValue.String);
858 
859 				// The empty string is a special case - we normally don't check
860 				// for the trailing null byte, in the case for the empty string
861 				// we do it explicitly, because there can't be keys in the
862 				// B+tree with a length of zero
863 				if (keySize == 0)
864 					keySize = 1;
865 			} else
866 				QUERY_RETURN_ERROR(B_ENTRY_NOT_FOUND);
867 		}
868 
869 		status = QueryPolicy::IndexIteratorFind(*iterator, Value(), keySize);
870 		if (Term<QueryPolicy>::fOp == OP_EQUAL && !fIsPattern)
871 			return status;
872 		else if (status == B_ENTRY_NOT_FOUND
873 			&& (fIsPattern || Term<QueryPolicy>::fOp == OP_GREATER_THAN
874 				|| Term<QueryPolicy>::fOp == OP_GREATER_THAN_OR_EQUAL))
875 			return B_OK;
876 
877 		QUERY_RETURN_ERROR(status);
878 	}
879 
880 	return B_OK;
881 }
882 
883 
884 template<typename QueryPolicy>
885 status_t
886 Equation<QueryPolicy>::GetNextMatching(Context* context,
887 	IndexIterator* iterator, struct dirent* dirent, size_t bufferSize)
888 {
889 	while (true) {
890 		union value<QueryPolicy> indexValue;
891 		size_t keyLength;
892 		Entry* entry = NULL;
893 
894 		status_t status = QueryPolicy::IndexIteratorGetNextEntry(iterator,
895 			&indexValue, &keyLength, (size_t)sizeof(indexValue), &entry);
896 		if (status != B_OK)
897 			return status;
898 
899 		// only compare against the index entry when this is the correct
900 		// index for the equation
901 		if (fHasIndex && !CompareTo((uint8*)&indexValue, keyLength)) {
902 			// They aren't equal? Let the operation decide what to do. Since
903 			// we always start at the beginning of the index (or the correct
904 			// position), only some needs to be stopped if the entry doesn't
905 			// fit.
906 			if (Term<QueryPolicy>::fOp == OP_LESS_THAN
907 				|| Term<QueryPolicy>::fOp == OP_LESS_THAN_OR_EQUAL
908 				|| (Term<QueryPolicy>::fOp == OP_EQUAL && !fIsPattern))
909 				return B_ENTRY_NOT_FOUND;
910 
911 			continue;
912 		}
913 
914 		// TODO: check user permissions here - but which one?!
915 		// we could filter out all those where we don't have
916 		// read access... (we should check for every parent
917 		// directory if the X_OK is allowed)
918 		// Although it's quite expensive to open all parents,
919 		// it's likely that the application that runs the
920 		// query will do something similar (and we don't have
921 		// to do it for root, either).
922 
923 		// go up in the tree until a &&-operator is found, and check if the
924 		// node matches with the rest of the expression - we don't have to
925 		// check ||-operators for that
926 		Term<QueryPolicy>* term = this;
927 		status = MATCH_OK;
928 
929 		if (!fHasIndex)
930 			status = Match(entry, QueryPolicy::EntryGetNode(entry));
931 
932 		while (term != NULL && status == MATCH_OK) {
933 			Operator<QueryPolicy>* parent
934 				= (Operator<QueryPolicy>*)term->Parent();
935 			if (parent == NULL)
936 				break;
937 
938 			if (parent->Op() == OP_AND) {
939 				// choose the other child of the parent
940 				Term<QueryPolicy>* other = parent->Right();
941 				if (other == term)
942 					other = parent->Left();
943 
944 				if (other == NULL) {
945 					QUERY_FATAL("&&-operator has only one child... "
946 						"(parent = %p)\n", parent);
947 					break;
948 				}
949 				status = other->Match(entry, QueryPolicy::EntryGetNode(entry));
950 				if (status < 0) {
951 					QUERY_REPORT_ERROR(status);
952 					status = NO_MATCH;
953 				}
954 			}
955 			term = (Term<QueryPolicy>*)parent;
956 		}
957 
958 		if (status == MATCH_OK) {
959 			ssize_t nameLength = QueryPolicy::EntryGetName(entry,
960 				dirent->d_name,
961 				(const char*)dirent + bufferSize - dirent->d_name);
962 			if (nameLength < 0)
963 				QUERY_RETURN_ERROR(nameLength);
964 
965 			dirent->d_dev = QueryPolicy::ContextGetVolumeID(context);
966 			dirent->d_ino = QueryPolicy::EntryGetNodeID(entry);
967 			dirent->d_pdev = dirent->d_dev;
968 			dirent->d_pino = QueryPolicy::EntryGetParentID(entry);
969 			dirent->d_reclen = sizeof(struct dirent) + strlen(dirent->d_name) + 1;
970 		}
971 
972 		if (status == MATCH_OK)
973 			return B_OK;
974 	}
975 	QUERY_RETURN_ERROR(B_ERROR);
976 }
977 
978 
979 template<typename QueryPolicy>
980 bool
981 Equation<QueryPolicy>::NeedsEntry()
982 {
983 	return strcmp(fAttribute, "name") == 0;
984 }
985 
986 
987 //	#pragma mark -
988 
989 
990 template<typename QueryPolicy>
991 Operator<QueryPolicy>::Operator(Term<QueryPolicy>* left, int8 op,
992 	Term<QueryPolicy>* right)
993 	:
994 	Term<QueryPolicy>(op),
995 	fLeft(left),
996 	fRight(right)
997 {
998 	if (left)
999 		left->SetParent(this);
1000 	if (right)
1001 		right->SetParent(this);
1002 }
1003 
1004 
1005 template<typename QueryPolicy>
1006 Operator<QueryPolicy>::~Operator()
1007 {
1008 	delete fLeft;
1009 	delete fRight;
1010 }
1011 
1012 
1013 template<typename QueryPolicy>
1014 status_t
1015 Operator<QueryPolicy>::Match(Entry* entry, Node* node, const char* attribute,
1016 	int32 type, const uint8* key, size_t size)
1017 {
1018 	if (Term<QueryPolicy>::fOp == OP_AND) {
1019 		status_t status = fLeft->Match(entry, node, attribute, type, key,
1020 			size);
1021 		if (status != MATCH_OK)
1022 			return status;
1023 
1024 		return fRight->Match(entry, node, attribute, type, key, size);
1025 	} else {
1026 		// choose the term with the better score for OP_OR
1027 		Term<QueryPolicy>* first;
1028 		Term<QueryPolicy>* second;
1029 		if (fRight->Score() > fLeft->Score()) {
1030 			first = fLeft;
1031 			second = fRight;
1032 		} else {
1033 			first = fRight;
1034 			second = fLeft;
1035 		}
1036 
1037 		status_t status = first->Match(entry, node, attribute, type, key,
1038 			size);
1039 		if (status != NO_MATCH)
1040 			return status;
1041 
1042 		return second->Match(entry, node, attribute, type, key, size);
1043 	}
1044 }
1045 
1046 
1047 template<typename QueryPolicy>
1048 void
1049 Operator<QueryPolicy>::Complement()
1050 {
1051 	if (Term<QueryPolicy>::fOp == OP_AND)
1052 		Term<QueryPolicy>::fOp = OP_OR;
1053 	else
1054 		Term<QueryPolicy>::fOp = OP_AND;
1055 
1056 	fLeft->Complement();
1057 	fRight->Complement();
1058 }
1059 
1060 
1061 template<typename QueryPolicy>
1062 void
1063 Operator<QueryPolicy>::CalculateScore(Index &index)
1064 {
1065 	fLeft->CalculateScore(index);
1066 	fRight->CalculateScore(index);
1067 }
1068 
1069 
1070 template<typename QueryPolicy>
1071 int32
1072 Operator<QueryPolicy>::Score() const
1073 {
1074 	if (Term<QueryPolicy>::fOp == OP_AND) {
1075 		// return the one with the better score
1076 		if (fRight->Score() > fLeft->Score())
1077 			return fRight->Score();
1078 
1079 		return fLeft->Score();
1080 	}
1081 
1082 	// for OP_OR, be honest, and return the one with the worse score
1083 	if (fRight->Score() < fLeft->Score())
1084 		return fRight->Score();
1085 
1086 	return fLeft->Score();
1087 }
1088 
1089 
1090 template<typename QueryPolicy>
1091 status_t
1092 Operator<QueryPolicy>::InitCheck()
1093 {
1094 	if ((Term<QueryPolicy>::fOp != OP_AND && Term<QueryPolicy>::fOp != OP_OR)
1095 		|| fLeft == NULL || fLeft->InitCheck() < B_OK
1096 		|| fRight == NULL || fRight->InitCheck() < B_OK)
1097 		return B_ERROR;
1098 
1099 	return B_OK;
1100 }
1101 
1102 
1103 template<typename QueryPolicy>
1104 bool
1105 Operator<QueryPolicy>::NeedsEntry()
1106 {
1107 	return ((fLeft && fLeft->NeedsEntry()) || (fRight && fRight->NeedsEntry()));
1108 }
1109 
1110 
1111 //	#pragma mark -
1112 
1113 #ifdef DEBUG_QUERY
1114 
1115 template<typename QueryPolicy>
1116 void
1117 Operator<QueryPolicy>::PrintToStream()
1118 {
1119 	QUERY_D(__out("( "));
1120 	if (fLeft != NULL)
1121 		fLeft->PrintToStream();
1122 
1123 	const char* op;
1124 	switch (Term<QueryPolicy>::fOp) {
1125 		case OP_OR: op = "OR"; break;
1126 		case OP_AND: op = "AND"; break;
1127 		default: op = "?"; break;
1128 	}
1129 	QUERY_D(__out(" %s ", op));
1130 
1131 	if (fRight != NULL)
1132 		fRight->PrintToStream();
1133 
1134 	QUERY_D(__out(" )"));
1135 }
1136 
1137 
1138 template<typename QueryPolicy>
1139 void
1140 Equation<QueryPolicy>::PrintToStream()
1141 {
1142 	const char* symbol = "???";
1143 	switch (Term<QueryPolicy>::fOp) {
1144 		case OP_EQUAL: symbol = "=="; break;
1145 		case OP_UNEQUAL: symbol = "!="; break;
1146 		case OP_GREATER_THAN: symbol = ">"; break;
1147 		case OP_GREATER_THAN_OR_EQUAL: symbol = ">="; break;
1148 		case OP_LESS_THAN: symbol = "<"; break;
1149 		case OP_LESS_THAN_OR_EQUAL: symbol = "<="; break;
1150 	}
1151 	QUERY_D(__out("[\"%s\" %s \"%s\"]", fAttribute, symbol, fString));
1152 }
1153 
1154 #endif	// DEBUG_QUERY
1155 
1156 //	#pragma mark -
1157 
1158 
1159 template<typename QueryPolicy>
1160 Expression<QueryPolicy>::Expression(char* expr)
1161 {
1162 	if (expr == NULL)
1163 		return;
1164 
1165 	fTerm = ParseOr(&expr);
1166 	if (fTerm != NULL && fTerm->InitCheck() < B_OK) {
1167 		QUERY_FATAL("Corrupt tree in expression!\n");
1168 		delete fTerm;
1169 		fTerm = NULL;
1170 	}
1171 	QUERY_D(if (fTerm != NULL) {
1172 		fTerm->PrintToStream();
1173 		QUERY_D(__out("\n"));
1174 		if (*expr != '\0')
1175 			PRINT(("Unexpected end of string: \"%s\"!\n", expr));
1176 	});
1177 	fPosition = expr;
1178 }
1179 
1180 
1181 template<typename QueryPolicy>
1182 Expression<QueryPolicy>::~Expression()
1183 {
1184 	delete fTerm;
1185 }
1186 
1187 
1188 template<typename QueryPolicy>
1189 Term<QueryPolicy>*
1190 Expression<QueryPolicy>::ParseEquation(char** expr)
1191 {
1192 	skipWhitespace(expr);
1193 
1194 	bool _not = false;
1195 	if (**expr == '!') {
1196 		skipWhitespace(expr, 1);
1197 		if (**expr != '(')
1198 			return NULL;
1199 
1200 		_not = true;
1201 	}
1202 
1203 	if (**expr == ')') {
1204 		// shouldn't be handled here
1205 		return NULL;
1206 	} else if (**expr == '(') {
1207 		skipWhitespace(expr, 1);
1208 
1209 		Term<QueryPolicy>* term = ParseOr(expr);
1210 
1211 		skipWhitespace(expr);
1212 
1213 		if (**expr != ')') {
1214 			delete term;
1215 			return NULL;
1216 		}
1217 
1218 		// If the term is negated, we just complement the tree, to get
1219 		// rid of the not, a.k.a. DeMorgan's Law.
1220 		if (_not)
1221 			term->Complement();
1222 
1223 		skipWhitespace(expr, 1);
1224 
1225 		return term;
1226 	}
1227 
1228 	Equation<QueryPolicy>* equation
1229 		= new(std::nothrow) Equation<QueryPolicy>(expr);
1230 	if (equation == NULL || equation->InitCheck() < B_OK) {
1231 		delete equation;
1232 		return NULL;
1233 	}
1234 	return equation;
1235 }
1236 
1237 
1238 template<typename QueryPolicy>
1239 Term<QueryPolicy>*
1240 Expression<QueryPolicy>::ParseAnd(char** expr)
1241 {
1242 	Term<QueryPolicy>* left = ParseEquation(expr);
1243 	if (left == NULL)
1244 		return NULL;
1245 
1246 	while (IsOperator(expr, '&')) {
1247 		Term<QueryPolicy>* right = ParseAnd(expr);
1248 		Term<QueryPolicy>* newParent = NULL;
1249 
1250 		if (right == NULL
1251 			|| (newParent = new(std::nothrow) Operator<QueryPolicy>(left,
1252 				OP_AND, right)) == NULL) {
1253 			delete left;
1254 			delete right;
1255 
1256 			return NULL;
1257 		}
1258 		left = newParent;
1259 	}
1260 
1261 	return left;
1262 }
1263 
1264 
1265 template<typename QueryPolicy>
1266 Term<QueryPolicy>*
1267 Expression<QueryPolicy>::ParseOr(char** expr)
1268 {
1269 	Term<QueryPolicy>* left = ParseAnd(expr);
1270 	if (left == NULL)
1271 		return NULL;
1272 
1273 	while (IsOperator(expr, '|')) {
1274 		Term<QueryPolicy>* right = ParseAnd(expr);
1275 		Term<QueryPolicy>* newParent = NULL;
1276 
1277 		if (right == NULL
1278 			|| (newParent = new(std::nothrow) Operator<QueryPolicy>(left, OP_OR,
1279 				right)) == NULL) {
1280 			delete left;
1281 			delete right;
1282 
1283 			return NULL;
1284 		}
1285 		left = newParent;
1286 	}
1287 
1288 	return left;
1289 }
1290 
1291 
1292 template<typename QueryPolicy>
1293 bool
1294 Expression<QueryPolicy>::IsOperator(char** expr, char op)
1295 {
1296 	char* string = *expr;
1297 
1298 	if (*string == op && *(string + 1) == op) {
1299 		*expr += 2;
1300 		return true;
1301 	}
1302 	return false;
1303 }
1304 
1305 
1306 template<typename QueryPolicy>
1307 status_t
1308 Expression<QueryPolicy>::InitCheck()
1309 {
1310 	if (fTerm == NULL)
1311 		return B_BAD_VALUE;
1312 
1313 	return B_OK;
1314 }
1315 
1316 
1317 //	#pragma mark -
1318 
1319 
1320 template<typename QueryPolicy>
1321 Query<QueryPolicy>::Query(Context* context, Expression<QueryPolicy>* expression,
1322 	uint32 flags, port_id port, uint32 token)
1323 	:
1324 	fContext(context),
1325 	fExpression(expression),
1326 	fCurrent(NULL),
1327 	fIterator(NULL),
1328 	fIndex(context),
1329 	fFlags(flags),
1330 	fPort(port),
1331 	fToken(token),
1332 	fNeedsEntry(false)
1333 {
1334 	// If the expression has a valid root pointer, the whole tree has
1335 	// already passed the sanity check, so that we don't have to check
1336 	// every pointer
1337 	if (context == NULL || expression == NULL || expression->Root() == NULL)
1338 		return;
1339 
1340 	// create index on the stack and delete it afterwards
1341 	fExpression->Root()->CalculateScore(fIndex);
1342 	QueryPolicy::IndexUnset(fIndex);
1343 
1344 	fNeedsEntry = fExpression->Root()->NeedsEntry();
1345 
1346 	Rewind();
1347 }
1348 
1349 
1350 template<typename QueryPolicy>
1351 Query<QueryPolicy>::~Query()
1352 {
1353 	delete fExpression;
1354 }
1355 
1356 
1357 template<typename QueryPolicy>
1358 /*static*/ status_t
1359 Query<QueryPolicy>::Create(Context* context, const char* queryString,
1360 	uint32 flags, port_id port, uint32 token, Query<QueryPolicy>*& _query)
1361 {
1362 	Expression<QueryPolicy>* expression
1363 		= new(std::nothrow) Expression<QueryPolicy>((char*)queryString);
1364 	if (expression == NULL)
1365 		QUERY_RETURN_ERROR(B_NO_MEMORY);
1366 
1367 	if (expression->InitCheck() != B_OK) {
1368 		QUERY_INFORM("Could not parse query \"%s\", stopped at: \"%s\"\n",
1369 			queryString, expression->Position());
1370 
1371 		delete expression;
1372 		QUERY_RETURN_ERROR(B_BAD_VALUE);
1373 	}
1374 
1375 	Query<QueryPolicy>* query = new(std::nothrow) Query<QueryPolicy>(context,
1376 		expression, flags, port, token);
1377 	if (query == NULL) {
1378 		delete expression;
1379 		QUERY_RETURN_ERROR(B_NO_MEMORY);
1380 	}
1381 
1382 	_query = query;
1383 	return B_OK;
1384 }
1385 
1386 
1387 template<typename QueryPolicy>
1388 status_t
1389 Query<QueryPolicy>::Rewind()
1390 {
1391 	// free previous stuff
1392 
1393 	fStack.MakeEmpty();
1394 
1395 	QueryPolicy::IndexIteratorDelete(fIterator);
1396 	fIterator = NULL;
1397 	fCurrent = NULL;
1398 
1399 	// put the whole expression on the stack
1400 
1401 	Stack<Term<QueryPolicy>*> stack;
1402 	stack.Push(fExpression->Root());
1403 
1404 	Term<QueryPolicy>* term;
1405 	while (stack.Pop(&term)) {
1406 		if (term->Op() < OP_EQUATION) {
1407 			Operator<QueryPolicy>* op = (Operator<QueryPolicy>*)term;
1408 
1409 			if (op->Op() == OP_OR) {
1410 				stack.Push(op->Left());
1411 				stack.Push(op->Right());
1412 			} else {
1413 				// For OP_AND, we can use the scoring system to decide which
1414 				// path to add
1415 				if (op->Right()->Score() > op->Left()->Score())
1416 					stack.Push(op->Right());
1417 				else
1418 					stack.Push(op->Left());
1419 			}
1420 		} else if (term->Op() == OP_EQUATION
1421 			|| fStack.Push((Equation<QueryPolicy>*)term) != B_OK)
1422 			QUERY_FATAL("Unknown term on stack or stack error\n");
1423 	}
1424 
1425 	return B_OK;
1426 }
1427 
1428 
1429 template<typename QueryPolicy>
1430 status_t
1431 Query<QueryPolicy>::GetNextEntry(struct dirent* dirent, size_t size)
1432 {
1433 	if (fIterator != NULL)
1434 		QueryPolicy::IndexIteratorResume(fIterator);
1435 
1436 	status_t error = _GetNextEntry(dirent, size);
1437 
1438 	if (fIterator != NULL)
1439 		QueryPolicy::IndexIteratorSuspend(fIterator);
1440 
1441 	return error;
1442 }
1443 
1444 
1445 template<typename QueryPolicy>
1446 void
1447 Query<QueryPolicy>::LiveUpdate(Entry* entry, Node* node, const char* attribute,
1448 	int32 type, const uint8* oldKey, size_t oldLength, const uint8* newKey,
1449 	size_t newLength)
1450 {
1451 	if (fPort < 0 || fExpression == NULL || attribute == NULL)
1452 		return;
1453 
1454 	// TODO: check if the attribute is part of the query at all...
1455 
1456 	// If no entry has been supplied, but the we need one for the evaluation
1457 	// (i.e. the "name" attribute is used), we invoke ourselves for all entries
1458 	// referring to the given node.
1459 	if (entry == NULL && fNeedsEntry) {
1460 		entry = QueryPolicy::NodeGetFirstReferrer(node);
1461 		while (entry) {
1462 			LiveUpdate(entry, node, attribute, type, oldKey, oldLength, newKey,
1463 				newLength);
1464 			entry = QueryPolicy::NodeGetNextReferrer(node, entry);
1465 		}
1466 		return;
1467 	}
1468 
1469 	status_t oldStatus = fExpression->Root()->Match(entry, node, attribute,
1470 		type, oldKey, oldLength);
1471 	status_t newStatus = fExpression->Root()->Match(entry, node, attribute,
1472 		type, newKey, newLength);
1473 
1474 	bool entryCreated = false;
1475 	bool stillInQuery = false;
1476 
1477 	if (oldStatus != MATCH_OK) {
1478 		if (newStatus != MATCH_OK) {
1479 			// nothing has changed
1480 			return;
1481 		}
1482 		entryCreated = true;
1483 	} else if (newStatus != MATCH_OK) {
1484 		// entry got removed
1485 		entryCreated = false;
1486 	} else if ((fFlags & B_ATTR_CHANGE_NOTIFICATION) != 0) {
1487 		// The entry stays in the query
1488 		stillInQuery = true;
1489 	} else
1490 		return;
1491 
1492 	// notify query listeners
1493 	status_t (*notify)(port_id, int32, dev_t, ino_t, const char*, ino_t);
1494 
1495 	if (stillInQuery)
1496 		notify = notify_query_attr_changed;
1497 	else if (entryCreated)
1498 		notify = notify_query_entry_created;
1499 	else
1500 		notify = notify_query_entry_removed;
1501 
1502 	if (entry != NULL) {
1503 		_SendEntryNotification(entry, notify);
1504 	} else {
1505 		entry = QueryPolicy::NodeGetFirstReferrer(node);
1506 		while (entry) {
1507 			_SendEntryNotification(entry, notify);
1508 			entry = QueryPolicy::NodeGetNextReferrer(node, entry);
1509 		}
1510 	}
1511 }
1512 
1513 
1514 template<typename QueryPolicy>
1515 void
1516 Query<QueryPolicy>::LiveUpdateRenameMove(Entry* entry, Node* node,
1517 	ino_t oldDirectoryID, const char* oldName, size_t oldLength,
1518 	ino_t newDirectoryID, const char* newName, size_t newLength)
1519 {
1520 	if (fPort < 0 || fExpression == NULL)
1521 		return;
1522 
1523 	// TODO: check if the attribute is part of the query at all...
1524 
1525 	status_t oldStatus = fExpression->Root()->Match(entry, node, "name",
1526 		B_STRING_TYPE, (const uint8*)oldName, oldLength);
1527 	status_t newStatus = fExpression->Root()->Match(entry, node, "name",
1528 		B_STRING_TYPE, (const uint8*)newName, newLength);
1529 
1530 	if (oldStatus != MATCH_OK || oldStatus != newStatus)
1531 		return;
1532 
1533 	// The entry stays in the query, notify query listeners about the rename
1534 	// or move
1535 
1536 	// We send a notification for the given entry, if any, or otherwise for
1537 	// all entries referring to the node;
1538 	if (entry != NULL) {
1539 		_SendEntryNotification(entry, notify_query_entry_removed);
1540 		_SendEntryNotification(entry, notify_query_entry_created);
1541 	} else {
1542 		entry = QueryPolicy::NodeGetFirstReferrer(node);
1543 		while (entry) {
1544 			_SendEntryNotification(entry, notify_query_entry_removed);
1545 			_SendEntryNotification(entry, notify_query_entry_created);
1546 			entry = QueryPolicy::NodeGetNextReferrer(node, entry);
1547 		}
1548 	}
1549 }
1550 
1551 
1552 template<typename QueryPolicy>
1553 status_t
1554 Query<QueryPolicy>::_GetNextEntry(struct dirent* dirent, size_t size)
1555 {
1556 	// If we don't have an equation to use yet/anymore, get a new one
1557 	// from the stack
1558 	while (true) {
1559 		if (fIterator == NULL) {
1560 			if (!fStack.Pop(&fCurrent)
1561 				|| fCurrent == NULL)
1562 				return B_ENTRY_NOT_FOUND;
1563 
1564 			status_t status = fCurrent->PrepareQuery(fContext, fIndex,
1565 				&fIterator, fFlags & B_QUERY_NON_INDEXED);
1566 			if (status == B_ENTRY_NOT_FOUND) {
1567 				// try next equation
1568 				continue;
1569 			}
1570 
1571 			if (status != B_OK)
1572 				return status;
1573 		}
1574 		if (fCurrent == NULL)
1575 			QUERY_RETURN_ERROR(B_ERROR);
1576 
1577 		status_t status = fCurrent->GetNextMatching(fContext, fIterator, dirent,
1578 			size);
1579 		if (status != B_OK) {
1580 			QueryPolicy::IndexIteratorDelete(fIterator);
1581 			fIterator = NULL;
1582 			fCurrent = NULL;
1583 		} else {
1584 			// only return if we have another entry
1585 			return B_OK;
1586 		}
1587 	}
1588 }
1589 
1590 
1591 template<typename QueryPolicy>
1592 void
1593 Query<QueryPolicy>::_SendEntryNotification(Entry* entry,
1594 	status_t (*notify)(port_id, int32, dev_t, ino_t, const char*, ino_t))
1595 {
1596 	char nameBuffer[QueryPolicy::kMaxFileNameLength];
1597 	const char* name = QueryPolicy::EntryGetNameNoCopy(entry, nameBuffer,
1598 		sizeof(nameBuffer));
1599 	if (name != NULL) {
1600 		notify(fPort, fToken, QueryPolicy::ContextGetVolumeID(fContext),
1601 			QueryPolicy::EntryGetParentID(entry), name,
1602 			QueryPolicy::EntryGetNodeID(entry));
1603 	}
1604 }
1605 
1606 
1607 }	// namespace QueryParser
1608 
1609 
1610 #endif	// _FILE_SYSTEMS_QUERY_PARSER_H
1611