#include "ssort.h"
#include "alloc.h"
#include "byte.h"
#include <stdlib.h>

/* this is a simple shellsort: we often get pre-sorted input in ftpcopy */
void mssort (char *base, uint32 n, uint32 elesize,
                  int (*cmp) (const void *, const void *))
{
  char *v;
  char buf[64];
  const uint32 dists[]={
    1, 5, 19, 41, 109, 209, 505, 929, 
    2161, 3905, 8929, 16001, 36289, 64769, 146305, 260609, 
    587521, 1045505, 2354689, 4188161, 9427969, 16764929, 37730305, 67084289,
    0
  };
  int distp;
  if (n==1) return;
  if (elesize<=sizeof(buf))
    v=buf;
  else {
    v=alloc(elesize);
    if (!v)
      return qsort(base,n,elesize,cmp);
  }
  for (distp=1;dists[distp] && dists[distp]<n;distp++)
    /* nothing */;
  distp--;
  /* if (distp) distp--; */
  while(1) {
    uint32 h=dists[distp];
    uint32 i,j;
/* printf("h=%lu\n",(unsigned long)h); */
/* puts(base); */
    for (i=h;i<n;i++) {
      j=i;
/* printf(" i=%lu\n",(unsigned long)i); */
      byte_copy(v,elesize,base+i*elesize);
      while (j>=h && cmp(base+(j-h)*elesize,v)>0) {
/* printf("  j=%lu\n",(unsigned long)j); */
	byte_copy(base+j*elesize,elesize,base+(j-h)*elesize);
	j-=h;
      }
      byte_copy(base+j*elesize,elesize,v);
    }
    if (!distp)
      break;
    distp--;
  }
  if (v!=buf)
    alloc_free(v);
}



syntax highlighted by Code2HTML, v. 0.9.1