138 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C
		
	
	
| #ifndef OID_ARRAY_H
 | |
| #define OID_ARRAY_H
 | |
| 
 | |
| #include "hash.h"
 | |
| 
 | |
| /**
 | |
|  * The API provides storage and manipulation of sets of object identifiers.
 | |
|  * The emphasis is on storage and processing efficiency, making them suitable
 | |
|  * for large lists. Note that the ordering of items is not preserved over some
 | |
|  * operations.
 | |
|  *
 | |
|  * Examples
 | |
|  * --------
 | |
|  * -----------------------------------------
 | |
|  * int print_callback(const struct object_id *oid,
 | |
|  * 		    void *data)
 | |
|  * {
 | |
|  * 	printf("%s\n", oid_to_hex(oid));
 | |
|  * 	return 0; // always continue
 | |
|  * }
 | |
|  *
 | |
|  * void some_func(void)
 | |
|  * {
 | |
|  *     struct oid_array hashes = OID_ARRAY_INIT;
 | |
|  *     struct object_id oid;
 | |
|  *
 | |
|  *     // Read objects into our set
 | |
|  *     while (read_object_from_stdin(oid.hash))
 | |
|  *         oid_array_append(&hashes, &oid);
 | |
|  *
 | |
|  *     // Check if some objects are in our set
 | |
|  *     while (read_object_from_stdin(oid.hash)) {
 | |
|  *         if (oid_array_lookup(&hashes, &oid) >= 0)
 | |
|  *             printf("it's in there!\n");
 | |
|  *
 | |
|  *          // Print the unique set of objects. We could also have
 | |
|  *          // avoided adding duplicate objects in the first place,
 | |
|  *          // but we would end up re-sorting the array repeatedly.
 | |
|  *          // Instead, this will sort once and then skip duplicates
 | |
|  *          // in linear time.
 | |
|  *
 | |
|  *         oid_array_for_each_unique(&hashes, print_callback, NULL);
 | |
|  *     }
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * A single array of object IDs. This should be initialized by assignment from
 | |
|  * `OID_ARRAY_INIT`. The `oid` member contains the actual data. The `nr` member
 | |
|  * contains the number of items in the set. The `alloc` and `sorted` members
 | |
|  * are used internally, and should not be needed by API callers.
 | |
|  */
 | |
| struct oid_array {
 | |
| 	struct object_id *oid;
 | |
| 	size_t nr;
 | |
| 	size_t alloc;
 | |
| 	int sorted;
 | |
| };
 | |
| 
 | |
| #define OID_ARRAY_INIT { 0 }
 | |
| 
 | |
| /**
 | |
|  * Add an item to the set. The object ID will be placed at the end of the array
 | |
|  * (but note that some operations below may lose this ordering).
 | |
|  */
 | |
| void oid_array_append(struct oid_array *array, const struct object_id *oid);
 | |
| 
 | |
| /**
 | |
|  * Perform a binary search of the array for a specific object ID. If found,
 | |
|  * returns the offset (in number of elements) of the object ID. If not found,
 | |
|  * returns a negative integer. If the array is not sorted, this function has
 | |
|  * the side effect of sorting it.
 | |
|  */
 | |
| int oid_array_lookup(struct oid_array *array, const struct object_id *oid);
 | |
| 
 | |
| /**
 | |
|  * Free all memory associated with the array and return it to the initial,
 | |
|  * empty state.
 | |
|  */
 | |
| void oid_array_clear(struct oid_array *array);
 | |
| 
 | |
| typedef int (*for_each_oid_fn)(const struct object_id *oid,
 | |
| 			       void *data);
 | |
| /**
 | |
|  * Iterate over each element of the list, executing the callback function for
 | |
|  * each one. Does not sort the list, so any custom hash order is retained.
 | |
|  * If the callback returns a non-zero value, the iteration ends immediately
 | |
|  * and the callback's return is propagated; otherwise, 0 is returned.
 | |
|  */
 | |
| int oid_array_for_each(struct oid_array *array,
 | |
| 		       for_each_oid_fn fn,
 | |
| 		       void *data);
 | |
| 
 | |
| /**
 | |
|  * Iterate over each unique element of the list in sorted order, but otherwise
 | |
|  * behave like `oid_array_for_each`. If the array is not sorted, this function
 | |
|  * has the side effect of sorting it.
 | |
|  */
 | |
| int oid_array_for_each_unique(struct oid_array *array,
 | |
| 			      for_each_oid_fn fn,
 | |
| 			      void *data);
 | |
| 
 | |
| /**
 | |
|  * Apply the callback function `want` to each entry in the array, retaining
 | |
|  * only the entries for which the function returns true. Preserve the order
 | |
|  * of the entries that are retained.
 | |
|  */
 | |
| void oid_array_filter(struct oid_array *array,
 | |
| 		      for_each_oid_fn want,
 | |
| 		      void *cbdata);
 | |
| 
 | |
| /**
 | |
|  * Sort the array in order of ascending object id.
 | |
|  */
 | |
| void oid_array_sort(struct oid_array *array);
 | |
| 
 | |
| /**
 | |
|  * Find the next unique oid in the array after position "cur".
 | |
|  * The array must be sorted for this to work. You can iterate
 | |
|  * over unique elements like this:
 | |
|  *
 | |
|  *   size_t i;
 | |
|  *   oid_array_sort(array);
 | |
|  *   for (i = 0; i < array->nr; i = oid_array_next_unique(array, i))
 | |
|  *	printf("%s", oid_to_hex(array->oids[i]);
 | |
|  *
 | |
|  * Non-unique iteration can just increment with "i++" to visit each element.
 | |
|  */
 | |
| static inline size_t oid_array_next_unique(struct oid_array *array, size_t cur)
 | |
| {
 | |
| 	do {
 | |
| 		cur++;
 | |
| 	} while (cur < array->nr &&
 | |
| 		 oideq(array->oid + cur, array->oid + cur - 1));
 | |
| 	return cur;
 | |
| }
 | |
| 
 | |
| #endif /* OID_ARRAY_H */
 |