1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 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
|
/*
* Copyright (C) 2019 Ernesto A. Fernández <ernesto.mnd.fernandez@gmail.com>
*/
#include <stdlib.h>
#include <stdio.h>
#include <apfs/types.h>
#include "apfsck.h"
#include "htable.h"
#include "super.h"
/**
* alloc_htable - Allocates and returns an empty hash table
*/
struct htable_entry **alloc_htable(void)
{
struct htable_entry **table;
table = calloc(HTABLE_BUCKETS, sizeof(*table));
if (!table)
system_error();
return table;
}
/**
* free_htable - Free a hash table and all its entries
* @table: the catalog table to free
* @free_entry: function that checks and frees an entry
*/
void free_htable(struct htable_entry **table,
void (*free_entry)(struct htable_entry *))
{
struct htable_entry *current;
struct htable_entry *next;
int i;
for (i = 0; i < HTABLE_BUCKETS; ++i) {
current = table[i];
while (current) {
next = current->h_next;
free_entry(current);
current = next;
}
}
free(table);
}
/**
* apply_on_htable - Apply a function to all entries in a hash table
* @table: the hash table
* @fn: function to apply
*/
void apply_on_htable(struct htable_entry **table, void (*fn)(struct htable_entry *))
{
struct htable_entry *current;
struct htable_entry *next;
int i;
for (i = 0; i < HTABLE_BUCKETS; ++i) {
current = table[i];
while (current) {
next = current->h_next;
fn(current);
current = next;
}
}
}
/**
* get_htable_entry - Find or create an entry in a hash table
* @id: id of the entry
* @size: size of the entry
* @table: the hash table
*
* Returns the entry, after creating it if necessary.
*/
struct htable_entry *get_htable_entry(u64 id, int size,
struct htable_entry **table)
{
int index = id % HTABLE_BUCKETS; /* Trivial hash function */
struct htable_entry **entry_p = table + index;
struct htable_entry *entry = *entry_p;
struct htable_entry *new;
/* In each linked list, entries are ordered by id */
while (entry) {
if (id == entry->h_id)
return entry;
if (id < entry->h_id)
break;
entry_p = &entry->h_next;
entry = *entry_p;
}
new = calloc(1, size);
if (!new)
system_error();
new->h_id = id;
new->h_next = entry;
*entry_p = new;
return new;
}
static void free_cnid(struct htable_entry *entry)
{
struct listed_cnid *cnid = (struct listed_cnid *)entry;
if (cnid->c_state & CNID_IN_SIBLING_LINK) {
if (cnid->c_state & ~CNID_IN_SIBLING_LINK)
report("Catalog", "sibling link oid reused elsewhere.");
}
free(entry);
}
/**
* free_cnid_table - Free the cnid hash table and all its entries
* @table: table to free
*
* Also performs some consistency checks that can only be done after the whole
* catalog has been parsed.
*/
void free_cnid_table(struct htable_entry **table)
{
/* No checks needed here, just call free() on each entry */
free_htable(table, free_cnid);
}
/**
* get_listed_cnid - Find or create a cnid structure in the cnid hash table
* @id: the cnid
*
* Returns the cnid structure, after creating it if necessary.
*/
struct listed_cnid *get_listed_cnid(u64 id)
{
struct htable_entry *entry;
entry = get_htable_entry(id, sizeof(struct listed_cnid),
vsb->v_cnid_table);
return (struct listed_cnid *)entry;
}
|