ICU-3138 add internal sort function to replace qsort

X-SVN-Rev: 12753
This commit is contained in:
Markus Scherer 2003-08-04 20:37:54 +00:00
parent 8e7d1112f4
commit d16be0ea7f
5 changed files with 313 additions and 1 deletions

View file

@ -66,7 +66,7 @@ uchar.o uprops.o propname.o ubidi.o ubidiwrt.o ubidiln.o ushape.o unames.o \
ucln_cmn.o uscript.o usc_impl.o umemstrm.o ucmp8.o uvector.o uvectr32.o digitlst.o \
brkiter.o brkdict.o ubrk.o dbbi.o dbbi_tbl.o \
rbbi.o rbbidata.o rbbinode.o rbbirb.o rbbiscan.o rbbisetb.o rbbistbl.o rbbitblb.o \
utrie.o uset.o cmemory.o caniter.o \
uarrsort.o utrie.o uset.o cmemory.o caniter.o \
unifilt.o unifunct.o uniset.o usetiter.o util.o uenum.o \
icuserv.o iculserv.o icunotif.o ustrenum.o \
uidna.o usprep.o punycode.o ucat.o

View file

@ -600,6 +600,14 @@ InputPath=.\unicode\strenum.h
# End Source File
# Begin Source File
SOURCE=.\uarrsort.c
# End Source File
# Begin Source File
SOURCE=.\uarrsort.h
# End Source File
# Begin Source File
SOURCE=.\ucmp8.c
# End Source File
# Begin Source File

View file

@ -363,6 +363,12 @@
Outputs="..\..\include\unicode\strenum.h"/>
</FileConfiguration>
</File>
<File
RelativePath=".\uarrsort.c">
</File>
<File
RelativePath=".\uarrsort.h">
</File>
<File
RelativePath=".\ucmp8.c">
</File>

View file

@ -0,0 +1,216 @@
/*
*******************************************************************************
*
* Copyright (C) 2003, International Business Machines
* Corporation and others. All Rights Reserved.
*
*******************************************************************************
* file name: uarrsort.c
* encoding: US-ASCII
* tab size: 8 (not used)
* indentation:4
*
* created on: 2003aug04
* created by: Markus W. Scherer
*
* Internal function for sorting arrays.
*/
#include "unicode/utypes.h"
#include "cmemory.h"
#include "uarrsort.h"
enum {
STACK_ITEM_SIZE=200
};
/* UComparator convenience implementations ---------------------------------- */
U_CAPI int32_t U_EXPORT2
uprv_uint16Comparator(const void *context, const void *left, const void *right) {
return (int32_t)*(const uint16_t *)left - (int32_t)*(const uint16_t *)right;
}
U_CAPI int32_t U_EXPORT2
uprv_int32Comparator(const void *context, const void *left, const void *right) {
return *(const int32_t *)left - *(const int32_t *)right;
}
U_CAPI int32_t U_EXPORT2
uprv_uint32Comparator(const void *context, const void *left, const void *right) {
uint32_t l=*(const uint32_t *)left, r=*(const uint32_t *)right;
/* compare directly because (l-r) would overflow the int32_t result */
if(l<r) {
return -1;
} else if(l==r) {
return 0;
} else /* l>r */ {
return 1;
}
}
/* Straight insertion sort from Knuth vol. III, pg. 81 ---------------------- */
static void
insertionSort(char *array, int32_t length, int32_t itemSize,
UComparator *cmp, const void *context, UErrorCode *pErrorCode) {
UAlignedMemory v[STACK_ITEM_SIZE/sizeof(UAlignedMemory)+1];
void *pv;
int32_t i, j;
/* allocate an intermediate item variable (v) */
if(itemSize<=STACK_ITEM_SIZE) {
pv=v;
} else {
pv=uprv_malloc(itemSize);
if(pv==NULL) {
*pErrorCode=U_MEMORY_ALLOCATION_ERROR;
return;
}
}
for(j=1; j<length; ++j) {
/* v=array[j] */
uprv_memcpy(pv, array+j*itemSize, itemSize);
for(i=j; i>0; --i) {
if(/* v>=array[i-1] */ cmp(context, pv, array+(i-1)*itemSize)>=0) {
break;
}
/* array[i]=array[i-1]; */
uprv_memcpy(array+i*itemSize, array+(i-1)*itemSize, itemSize);
}
if(i!=j) {
/* array[i]=v; */
uprv_memcpy(array+i*itemSize, pv, itemSize);
}
}
if(pv!=v) {
uprv_free(pv);
}
}
/* QuickSort ---------------------------------------------------------------- */
/*
* This implementation is semi-recursive:
* It recurses for the smaller sub-array to shorten the recursion depth,
* and loops for the larger sub-array.
*/
static void
subQuickSort(char *array, int32_t start, int32_t limit, int32_t itemSize,
UComparator *cmp, const void *context,
void *px, void *pw) {
int32_t left, right;
/* start and left are inclusive, limit and right are exclusive */
do {
left=start;
right=limit;
/* x=array[middle] */
uprv_memcpy(px, array+((start+limit)/2)*itemSize, itemSize);
do {
while(/* array[left]<x */
cmp(context, array+left*itemSize, px)<0
) {
++left;
}
while(/* x<array[right-1] */
cmp(context, px, array+(right-1)*itemSize)<0
) {
--right;
}
/* swap array[left] and array[right-1] via w; ++left; --right */
if(left<right) {
--right;
if(left<right) {
uprv_memcpy(pw, array+left*itemSize, itemSize);
uprv_memcpy(array+left*itemSize, array+right*itemSize, itemSize);
uprv_memcpy(array+right*itemSize, pw, itemSize);
}
++left;
}
} while(left<right);
/* sort sub-arrays */
if((right-start)<(limit-left)) {
/* sort [start..right[ */
if(start<(right-1)) {
subQuickSort(array, start, right, itemSize, cmp, context, px, pw);
}
/* sort [left..limit[ */
start=left;
} else {
/* sort [left..limit[ */
if(left<(limit-1)) {
subQuickSort(array, left, limit, itemSize, cmp, context, px, pw);
}
/* sort [start..right[ */
limit=right;
}
} while(start<(limit-1));
}
static void
quickSort(char *array, int32_t length, int32_t itemSize,
UComparator *cmp, const void *context, UErrorCode *pErrorCode) {
UAlignedMemory xw[(2*STACK_ITEM_SIZE)/sizeof(UAlignedMemory)+1];
void *p;
/* allocate two intermediate item variables (x and w) */
if(itemSize<=STACK_ITEM_SIZE) {
p=xw;
} else {
p=uprv_malloc(2*itemSize);
if(p==NULL) {
*pErrorCode=U_MEMORY_ALLOCATION_ERROR;
return;
}
}
subQuickSort(array, 0, length, itemSize,
cmp, context, p, (char *)p+itemSize);
if(p!=xw) {
uprv_free(p);
}
}
/* uprv_sortArray() API ----------------------------------------------------- */
/*
* Check arguments, select an appropriate implementation,
* cast the array to char * so that array+i*itemSize works.
*/
U_CAPI void U_EXPORT2
uprv_sortArray(void *array, int32_t length, int32_t itemSize,
UComparator *cmp, const void *context,
UBool sortStable, UErrorCode *pErrorCode) {
if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
return;
}
if((length>0 && array==NULL) || length<0 || itemSize<=0 || cmp==NULL) {
*pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
return;
}
if(length<=1) {
return;
} else if(length<=10 || sortStable) {
insertionSort((char *)array, length, itemSize, cmp, context, pErrorCode);
} else {
/* quickSort */
quickSort((char *)array, length, itemSize, cmp, context, pErrorCode);
}
}

View file

@ -0,0 +1,82 @@
/*
*******************************************************************************
*
* Copyright (C) 2003, International Business Machines
* Corporation and others. All Rights Reserved.
*
*******************************************************************************
* file name: uarrsort.h
* encoding: US-ASCII
* tab size: 8 (not used)
* indentation:4
*
* created on: 2003aug04
* created by: Markus W. Scherer
*
* Internal function for sorting arrays.
*/
#ifndef __UARRSORT_H__
#define __UARRSORT_H__
#include "unicode/utypes.h"
/**
* Function type for comparing two items as part of sorting an array or similar.
* Callback function for uprv_sortArray().
*
* @param context Application-specific pointer, passed through by uprv_sortArray().
* @param left Pointer to the "left" item.
* @param right Pointer to the "right" item.
* @return 32-bit signed integer comparison result:
* <0 if left<right
* ==0 if left==right
* >0 if left>right
*
* @internal
*/
typedef int32_t U_CALLCONV
UComparator(const void *context, const void *left, const void *right);
/**
* Array sorting function.
* Uses a UComparator for comparing array items to each other, and simple
* memory copying to move items.
*
* @param array The array to be sorted.
* @param length The number of items in the array.
* @param itemSize The size in bytes of each array item.
* @param cmp UComparator function used to compare two items each.
* @param context Application-specific pointer, passed through to the UComparator.
* @param sortStable If true, a stable sorting algorithm must be used.
* @param pErrorCode ICU in/out UErrorCode parameter.
*
* @internal
*/
U_CAPI void U_EXPORT2
uprv_sortArray(void *array, int32_t length, int32_t itemSize,
UComparator *cmp, const void *context,
UBool sortStable, UErrorCode *pErrorCode);
/**
* Convenience UComparator implementation for uint16_t arrays.
* @internal
*/
U_CAPI int32_t U_EXPORT2
uprv_uint16Comparator(const void *context, const void *left, const void *right);
/**
* Convenience UComparator implementation for int32_t arrays.
* @internal
*/
U_CAPI int32_t U_EXPORT2
uprv_int32Comparator(const void *context, const void *left, const void *right);
/**
* Convenience UComparator implementation for uint32_t arrays.
* @internal
*/
U_CAPI int32_t U_EXPORT2
uprv_uint32Comparator(const void *context, const void *left, const void *right);
#endif