Auto merged
2003/09/24 14:13:09+01:00 cantab.net!aia21 Fix a silly bug in attrib.c::ntfs_external_attr_find() which caused us to always return the same attribute during an enumeration as soon as the end of the attributes to be enumerated was reached thus causing the program doing the enumeration to hang in a tight loop. NOTE: There is still work to do as ctx->al_entry is set to NULL instead of to the position in the attribute list at which to insert. (Logical change 1.182)edge.strict_endians
							parent
							
								
									2bd3a2dd44
								
							
						
					
					
						commit
						e9bce127fa
					
				|  | @ -1408,6 +1408,9 @@ static int ntfs_attr_find(const ATTR_TYPES type, const uchar_t *name, | ||||||
|  * @ctx->base_ntfs_ino->attr_list at which the new attribute's attribute list |  * @ctx->base_ntfs_ino->attr_list at which the new attribute's attribute list | ||||||
|  * entry should be inserted. |  * entry should be inserted. | ||||||
|  * |  * | ||||||
|  |  * FIXME: This is how it should be but unfortunately the current code sets | ||||||
|  |  *        @ctx->al_entry to NULL, so beware! (AIA) | ||||||
|  |  * | ||||||
|  * The following error codes are defined: |  * The following error codes are defined: | ||||||
|  *	ENOENT	Attribute not found, not an error as such. |  *	ENOENT	Attribute not found, not an error as such. | ||||||
|  *	EINVAL	Invalid arguments. |  *	EINVAL	Invalid arguments. | ||||||
|  | @ -1711,27 +1714,37 @@ do_next_attr: | ||||||
| 	return -1; | 	return -1; | ||||||
| not_found: | not_found: | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Seek to the end of the base mft record, i.e. when we return false, | 	 * The attribute wasn't found.  Before we return, we want to ensure | ||||||
| 	 * ctx->mrec and ctx->attr indicate where the attribute should be | 	 * ctx->mrec and ctx->attr indicate the position at which the attribute | ||||||
| 	 * inserted into the attribute record. | 	 * should be inserted in the base mft record. | ||||||
| 	 * And of course ctx->al_entry points to the end of the attribute |  | ||||||
| 	 * list inside ctx->base_ntfs_ino->attr_list. |  | ||||||
| 	 * |  | ||||||
| 	 * FIXME: Do we really want to do this here? Think about it... (AIA) |  | ||||||
| 	 */ | 	 */ | ||||||
|  | 	/* Rewind the current search so ntfs_attr_find() is happy. */ | ||||||
| 	ntfs_attr_reinit_search_ctx(ctx); | 	ntfs_attr_reinit_search_ctx(ctx); | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * If we were enumerating and reached the end, we can't just use !@type | 	 * If we were enumerating and reached the end, we can't just use @type | ||||||
| 	 * because that would return the first attribute instead of the last | 	 * because that would return the first attribute instead of the last | ||||||
| 	 * one. Thus we just change @type to AT_END which causes | 	 * one. Thus we just use AT_END which causes ntfs_attr_find() to seek | ||||||
| 	 * ntfs_attr_find() to seek to the end. We also do the same when an | 	 * to the end. | ||||||
| 	 * attribute extent was searched for (i.e. @lowest_vcn != 0), as we |  | ||||||
| 	 * otherwise rewind the search back to the first extent and we get |  | ||||||
| 	 * that extent returned twice during a search for all extents. |  | ||||||
| 	 */ | 	 */ | ||||||
| 	if (!type || lowest_vcn) | 	if (!type) | ||||||
| 		type = AT_END; | 		return ntfs_attr_find(AT_END, name, name_len, ic, val, val_len, | ||||||
| 	return ntfs_attr_find(type, name, name_len, ic, val, val_len, ctx); | 				ctx); | ||||||
|  | 	/*
 | ||||||
|  | 	 * In case there are multiple matches in the base mft record, need to | ||||||
|  | 	 * keep enumerating until we get an attribute not found response (or | ||||||
|  | 	 * another error), otherwise we would keep returning the same attribute | ||||||
|  | 	 * over and over again and all programs using us for enumeration would | ||||||
|  | 	 * lock up in a tight loop. | ||||||
|  | 	 */ | ||||||
|  | 	{ | ||||||
|  | 		int ret; | ||||||
|  | 
 | ||||||
|  | 		do { | ||||||
|  | 			ret = ntfs_attr_find(type, name, name_len, ic, val, | ||||||
|  | 					val_len, ctx); | ||||||
|  | 		} while (!ret); | ||||||
|  | 		return ret; | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -1785,6 +1798,9 @@ not_found: | ||||||
|  * the position within @ctx->base_ntfs_ino->attr_list at which the new |  * the position within @ctx->base_ntfs_ino->attr_list at which the new | ||||||
|  * attribute's attribute list entry should be inserted. |  * attribute's attribute list entry should be inserted. | ||||||
|  * |  * | ||||||
|  |  * FIXME: This is how it should be but unfortunately the current code sets | ||||||
|  |  *        @ctx->al_entry to NULL, so beware! (AIA) | ||||||
|  |  * | ||||||
|  * The following error codes are defined: |  * The following error codes are defined: | ||||||
|  *	ENOENT	Attribute not found, not an error as such. |  *	ENOENT	Attribute not found, not an error as such. | ||||||
|  *	EINVAL	Invalid arguments. |  *	EINVAL	Invalid arguments. | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue