#include <stdio.h>

//K&R Chapt5, Section   4
#if 0
#define ALLOCSIZE 10000
static char allocbuf[ALLOCSIZE];    /*storage for alloc*/
static char *allocp = allocbuf;    /*next free position*/

char *alloc(int n)
{
    if(allocbuf+ALLOCSIZE - allocp >= n) {
        allocp += n;
        return alloc - n;
    } else
        return 0;
}

void afree(char *p)
{
    if (p >= allocbuf && p<allocbuf + ALLOCSIZE)
        allocp = p;
}
#endif


//K&R Chapt8, Section 7
#define NULL 0
typedef long Align;/*for alignment to long boundary*/
union header {
    struct {
        union header *ptr; /*next block if on free list*/
        unsigned size; /*size of this block*/
    } s;
    Align x;
};

typedef union header Header;


static Header base;
static Header *freep = NULL;



void kr_free(void *ap)
{
    Header *bp,*p;
    bp = (Header *)ap -1; /* point to block header */
    printf("kr_free: bp 0x%x, size %d\n",bp,bp->s.size);
    for(p=freep;!(bp>p && bp< p->s.ptr);p=p->s.ptr)
        if(p>=p->s.ptr && (bp>p || bp<p->s.ptr))
            break;    /* freed block at start or end of arena*/
    if (bp+bp->s.size==p->s.ptr) {    /* join to upper nbr */
        bp->s.size += p->s.ptr->s.size;
        bp->s.ptr = p->s.ptr->s.ptr;
    } else
        bp->s.ptr = p->s.ptr;
    if (p+p->s.size == bp) {     /* join to lower nbr */
        p->s.size += bp->s.size;
        p->s.ptr = bp->s.ptr;
    } else
        p->s.ptr = bp;
    freep = p;
}


#define NALLOC 1024    /* minimum #units to request */
static Header *morecore(unsigned nu)
{
    char *cp;
    Header *up;
    if(nu < NALLOC)
        nu = NALLOC;

    cp = sbrk(nu * sizeof(Header));
    printf("morecore: cp 0x%x, size %d\n",cp,nu * sizeof(Header));
    if(cp == (char *)-1)    /* no space at all*/
        return NULL;
    up = (Header *)cp;
    up->s.size = nu;
    kr_free((void *)(up+1));
    return freep;
}


void *kr_malloc(unsigned nbytes)
{
    Header *p, *prevp;
    unsigned nunits;
    nunits = (nbytes+sizeof(Header)-1)/sizeof(Header) + 1;
   printf("kr_malloc: nbytes %d, nunits %d\n",nbytes,nunits);    
    if((prevp = freep) == NULL) { /* no free list */
        base.s.ptr = freep = prevp = &base;
        base.s.size = 0;
    }
    for(p = prevp->s.ptr; ;prevp = p, p= p->s.ptr) {
        if(p->s.size >= nunits) { /* big enough */
            if (p->s.size == nunits)  /* exactly */
                prevp->s.ptr = p->s.ptr;
            else {
                p->s.size -= nunits;
                p += p->s.size;
                p->s.size = nunits;
            }
            freep = prevp;
            printf("kr_malloc: p 0x%x, size %d\n",p,p->s.size);
            return (void*)(p+1);
        }
        if (p== freep) /* wrapped around free list */
            if ((p = morecore(nunits)) == NULL) {
                printf("kr_malloc: no free space!!!\n");
                return NULL; /* none left */
            }
    }
}


void kr_dump(void)
{ static int i=0;
    Header *p, *prevp;
	prevp = freep;
	i++;
	for(p = prevp->s.ptr; ;prevp = p, p= p->s.ptr) {
		if(p== freep) {
		 	return;
		}
		printf("kr_dump: %d: p 0x%x, size %d\n", i, p, p->s.size);

	}
}

void main(void)
{
   void * na,*nb,*nc,*nd,*ne, *nf;
   printf("size of Header  is %d\n", sizeof(Header));
   na=kr_malloc(16);
   kr_dump();
   kr_free(na);
   kr_dump();
   nb=kr_malloc(48);
   kr_dump();
   nc=kr_malloc(60);
   kr_dump();
   nd=kr_malloc(128);
   kr_dump();
   kr_free(nb);
   kr_dump();
   ne=kr_malloc(512);
   kr_dump();
   kr_free(nd);
   kr_dump();
   nf=kr_malloc(256);
   kr_dump();
   kr_free(nc);
   kr_dump();
   kr_free(nf);
   kr_dump();
   kr_free(ne);
   kr_dump();
}