shithub: purgatorio

ref: fa9f4c5c6852ce467644f610e77504664cb86af4
dir: /liblogfs/extentlist.c/

View raw version
#include "logfsos.h"
#include "logfs.h"
#include "local.h"

typedef struct ExtentNode ExtentNode;

struct ExtentNode {
	Extent e;
	ExtentNode *next;
};

struct ExtentList {
	ExtentNode *head;
};

char *
logfsextentlistnew(ExtentList **ep)
{
	ExtentList *l;
	l = logfsrealloc(nil, sizeof(*l));
	if(l == nil)
		return Enomem;
	*ep = l;
	return  nil;
}

void
logfsextentlistreset(ExtentList *l)
{
	ExtentNode *n;
	n = l->head;
	while(n) {
		ExtentNode *next;
		next = n->next;
		logfsfreemem(n);
		n = next;
	}
	l->head = nil;
}

void
logfsextentlistfree(ExtentList **lp)
{
	ExtentList *l;
	if(lp == nil || (l = *lp) == nil)
		return;
	logfsextentlistreset(l);
	logfsfreemem(l);
	*lp = nil;
}

char *
logfsextentlistinsert(ExtentList *l, Extent *add, Extent **new)
{
	ExtentNode *old, *prev;
	ExtentNode *saved = nil;

	if(l == nil)
		return "nil extentlist";

	/* initially a's extents are non-empty, disjoint and sorted */
	old = l->head;
	prev = nil;
	while(old) {
		ExtentNode *next = old->next;
		if(add->max <= old->e.min)
			break;
		if(old->e.min < add->max && add->min < old->e.max) {	/* they intersect */
			if(add->min <= old->e.min) {
				/* add overlaps front of old */
				if(add->max < old->e.max) {
					int trimmed;
					/* but doesn't overlap end */
					/* retain tail of old */
					if(saved == nil){
						saved = logfsrealloc(nil, sizeof(*saved));
						if(saved == nil)
							return Enomem;
					}
					trimmed = add->max - old->e.min;
					old->e.min += trimmed;
					old->e.flashaddr += trimmed;
					/* can't touch any further extents, so... */
					break;
				}
				/* add.min ≤ old.min < old.max ≤ add.max ⇒ add completely covers old */
				/* delete old */
				if(prev)
					prev->next = next;
				else
					l->head = next;
				/* stash the deleted extent, so we can reuse it */
				if(saved == nil)
					saved = old;
				else
					logfsfreemem(old);
				old = next;
				continue;
			}
			else {
				/* add.min > old.min, so overlaps end of old or splits it */
				if(add->max < old->e.max) {	/* add inside old, splitting it */
					ExtentNode *frag;
					/*
					 * will need at most two add extents, so ensure
					 * enough store exists before changing data structures
					 */
					if(saved == nil)
						saved = logfsrealloc(nil, sizeof(*saved));
					frag = logfsrealloc(nil, sizeof(*frag));
					if(saved == nil || frag == nil)
						return Enomem;
					frag->next = next;
					old->next = frag;
					frag->e.min = add->max;
					frag->e.max = old->e.max;
					frag->e.flashaddr = old->e.flashaddr + (add->max - old->e.min);
					old->e.max = add->min;
					prev = old;
					break;
				}
				else {
					/*
					 * will need at most one add extent, so create one
					 * now before changing data structures
					 */
					if(saved == nil){
						saved = logfsrealloc(nil, sizeof(*saved));
						if(saved == nil)
							return Enomem;
					}
					old->e.max = add->min;		/* retain start of old */
				}
				/* old.max <= add.max ⇒ add covers tail of old */
			}
		}
		prev = old;
		old = next;
	}
	/*
	 * if here, and saved == nil, then there was no overlap
	 */
	if(saved == nil){
		saved = logfsrealloc(nil, sizeof(*saved));
		if(saved == nil)
			return Enomem;
	}
	saved->e = *add;
	if(prev) {
		saved->next = prev->next;
		prev->next = saved;
	}
	else {
		saved->next = l->head;
		l->head = saved;
	}
	if(new)
		*new = &saved->e;
	return nil;
}

Extent *
logfsextentlistmatch(ExtentList *l, Extent *e)
{
	ExtentNode *m;
	u32int flashmax;

	if(l == nil)
		return nil;

	flashmax = e->flashaddr + (e->max - e->min);

	for(m = l->head; m; m = m->next) {
		u32int l = m->e.max - m->e.min;
		if(e->min < m->e.max && m->e.min < e->max	/* they intersect */
			&& m->e.flashaddr < flashmax && e->flashaddr < m->e.flashaddr + l) /* the store intersects */
			return &(m->e);
	}
	return nil;
}

int
logfsextentlistmatchall(ExtentList *l, int (*func)(void *magic, Extent *), void *magic, Extent *e)
{
	ExtentNode *m;
	u32int flashmax;

	if(l == nil)
		return 1;

	flashmax = e->flashaddr + (e->max - e->min);

	for(m = l->head; m; m = m->next) {
		u32int l;
		if(m->e.min >= e->max)
			return 1;
		l = m->e.max - m->e.min;
		if(e->min < m->e.max	/* they intersect */
			&& m->e.flashaddr < flashmax && e->flashaddr < m->e.flashaddr + l) {
			/* the store intersects */
			int rv = (*func)(magic, &(m->e));
			if(rv <= 0)
				return rv;
		}
	}
	return 1;
}

int
logfsextentlistwalk(ExtentList *l, int (*func)(void *magic, Extent *, int hole), void *magic)
{
	ExtentNode *n;
	u32int last = 0;
	if(l == nil)
		return 1;
	for(n = l->head; n; n = n->next) {
		int rv;
		if(last < n->e.min) {
			Extent hole;
			hole.min = last;
			hole.max = n->e.min;
			hole.flashaddr = ~0;
			rv = (*func)(magic, &hole, 1);
			if(rv <= 0)
				return rv;
		}
		rv = (*func)(magic, &n->e, 0);
		if(rv <= 0)
			return rv;
		last = n->e.max;
	}
	return 1;
}

int
logfsextentlistwalkrange(ExtentList *l, int (*func)(void *magic, u32int baseoffset, u32int limitoffset, Extent *, u32int extentoffset), void *magic, u32int base, u32int limit)
{
	ExtentNode *n;
	u32int last = 0;
	if(l == nil)
		return 1;
	for(n = l->head; n; n = n->next) {
		Extent hole;
		Extent *e;
		if(last < n->e.min) {
			hole.min = last;
			hole.max = n->e.min;
			e = &hole;
		}
		else {
		again:
			e = &n->e;
		}
		if(e->min >= limit)
			return 1;
//print("walkrange %ud .. %ud\n", e->min, e->max);
		if(e->max > base) {
			ulong rangebase, rangelimit, extentoffset;
			int rv;
			rangebase = e->min;
			if(rangebase < base) {
				extentoffset = base - rangebase;
				rangebase += extentoffset;
			}
			else
				extentoffset = 0;
			rangelimit = e->max;
			if(rangelimit > limit)
				rangelimit = limit;
			rv = (*func)(magic, rangebase - base, rangelimit - base, e == &hole ? nil : e, extentoffset);
			if(rv <= 0)
				return rv;
		}
		last = e->max;
		if(e == &hole)
			goto again;
	}
	return 1;
}