Added sort functions for ADT Arrays, see sorting.inc
--HG-- extra : convert_revision : svn%3A39bc706e-5318-0410-9160-8a85361fbb7c/trunk%401707
This commit is contained in:
parent
4aa9ae379f
commit
fa3f1991ff
@ -32,6 +32,8 @@
|
|||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
extern HandleType_t htCellArray;
|
||||||
|
|
||||||
class CellArray
|
class CellArray
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
@ -149,6 +151,11 @@ public:
|
|||||||
return array;
|
return array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cell_t *base()
|
||||||
|
{
|
||||||
|
return m_Data;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool GrowIfNeeded(size_t count)
|
bool GrowIfNeeded(size_t count)
|
||||||
{
|
{
|
||||||
|
@ -33,6 +33,8 @@
|
|||||||
#include <IHandleSys.h>
|
#include <IHandleSys.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include "HandleSys.h"
|
||||||
|
#include "CellArray.h"
|
||||||
|
|
||||||
/***********************************
|
/***********************************
|
||||||
* About the double array hack *
|
* About the double array hack *
|
||||||
@ -373,6 +375,139 @@ static cell_t sm_SortCustom2D(IPluginContext *pContext, const cell_t *params)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum SortType
|
||||||
|
{
|
||||||
|
Sort_Integer = 0,
|
||||||
|
Sort_Float,
|
||||||
|
Sort_String,
|
||||||
|
};
|
||||||
|
|
||||||
|
int sort_adtarray_strings_asc(const void *str1, const void *str2)
|
||||||
|
{
|
||||||
|
return strcmp((char *) str1, (char *) str2);
|
||||||
|
}
|
||||||
|
|
||||||
|
int sort_adtarray_strings_desc(const void *str1, const void *str2)
|
||||||
|
{
|
||||||
|
return strcmp((char *) str2, (char *) str1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static cell_t sm_SortADTArray(IPluginContext *pContext, const cell_t *params)
|
||||||
|
{
|
||||||
|
CellArray *cArray;
|
||||||
|
HandleError err;
|
||||||
|
HandleSecurity sec(pContext->GetIdentity(), g_pCoreIdent);
|
||||||
|
|
||||||
|
if ((err = g_HandleSys.ReadHandle(params[1], htCellArray, &sec, (void **)&cArray))
|
||||||
|
!= HandleError_None)
|
||||||
|
{
|
||||||
|
return pContext->ThrowNativeError("Invalid Handle %x (error: %d)", params[1], err);
|
||||||
|
}
|
||||||
|
|
||||||
|
cell_t order = params[2];
|
||||||
|
cell_t type = params[3];
|
||||||
|
size_t arraysize = cArray->size();
|
||||||
|
size_t blocksize = cArray->blocksize();
|
||||||
|
cell_t *array = cArray->base();
|
||||||
|
|
||||||
|
if (type == Sort_Integer)
|
||||||
|
{
|
||||||
|
if (order == Sort_Ascending)
|
||||||
|
{
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_ints_asc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_ints_desc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (type == Sort_Float)
|
||||||
|
{
|
||||||
|
if (type == Sort_Ascending)
|
||||||
|
{
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_floats_asc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_floats_desc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (type == Sort_String)
|
||||||
|
{
|
||||||
|
if (type == Sort_Ascending)
|
||||||
|
{
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_adtarray_strings_asc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_adtarray_strings_desc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct sort_infoADT
|
||||||
|
{
|
||||||
|
IPluginFunction *pFunc;
|
||||||
|
cell_t *array_base;
|
||||||
|
cell_t array_bsize;
|
||||||
|
Handle_t array_hndl;
|
||||||
|
Handle_t hndl;
|
||||||
|
};
|
||||||
|
|
||||||
|
sort_infoADT g_SortInfoADT;
|
||||||
|
|
||||||
|
int sort_adtarray_custom(const void *elem1, const void *elem2)
|
||||||
|
{
|
||||||
|
cell_t result = 0;
|
||||||
|
IPluginFunction *pf = g_SortInfoADT.pFunc;
|
||||||
|
pf->PushCell(((cell_t) ((cell_t *) elem1 - g_SortInfoADT.array_base)) / g_SortInfoADT.array_bsize);
|
||||||
|
pf->PushCell(((cell_t) ((cell_t *) elem2 - g_SortInfoADT.array_base)) / g_SortInfoADT.array_bsize);
|
||||||
|
pf->PushCell(g_SortInfoADT.array_hndl);
|
||||||
|
pf->PushCell(g_SortInfoADT.hndl);
|
||||||
|
pf->Execute(&result);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static cell_t sm_SortADTArrayCustom(IPluginContext *pContext, const cell_t *params)
|
||||||
|
{
|
||||||
|
CellArray *cArray;
|
||||||
|
HandleError err;
|
||||||
|
HandleSecurity sec(pContext->GetIdentity(), g_pCoreIdent);
|
||||||
|
|
||||||
|
if ((err = g_HandleSys.ReadHandle(params[1], htCellArray, &sec, (void **)&cArray))
|
||||||
|
!= HandleError_None)
|
||||||
|
{
|
||||||
|
return pContext->ThrowNativeError("Invalid Handle %x (error: %d)", params[1], err);
|
||||||
|
}
|
||||||
|
|
||||||
|
IPluginFunction *pFunction = pContext->GetFunctionById(params[2]);
|
||||||
|
if (!pFunction)
|
||||||
|
{
|
||||||
|
return pContext->ThrowNativeError("Function %x is not a valid function", params[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t arraysize = cArray->size();
|
||||||
|
size_t blocksize = cArray->blocksize();
|
||||||
|
cell_t *array = cArray->base();
|
||||||
|
|
||||||
|
sort_infoADT oldinfo = g_SortInfoADT;
|
||||||
|
|
||||||
|
g_SortInfoADT.pFunc = pFunction;
|
||||||
|
g_SortInfoADT.array_base = array;
|
||||||
|
g_SortInfoADT.array_bsize = (cell_t) blocksize;
|
||||||
|
g_SortInfoADT.array_hndl = params[1];
|
||||||
|
g_SortInfoADT.hndl = params[3];
|
||||||
|
|
||||||
|
qsort(array, arraysize, blocksize * sizeof(cell_t), sort_adtarray_custom);
|
||||||
|
|
||||||
|
g_SortInfoADT = oldinfo;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
REGISTER_NATIVES(sortNatives)
|
REGISTER_NATIVES(sortNatives)
|
||||||
{
|
{
|
||||||
{"SortIntegers", sm_SortIntegers},
|
{"SortIntegers", sm_SortIntegers},
|
||||||
@ -380,5 +515,7 @@ REGISTER_NATIVES(sortNatives)
|
|||||||
{"SortStrings", sm_SortStrings},
|
{"SortStrings", sm_SortStrings},
|
||||||
{"SortCustom1D", sm_SortCustom1D},
|
{"SortCustom1D", sm_SortCustom1D},
|
||||||
{"SortCustom2D", sm_SortCustom2D},
|
{"SortCustom2D", sm_SortCustom2D},
|
||||||
|
{"SortADTArray", sm_SortADTArray},
|
||||||
|
{"SortADTArrayCustom", sm_SortADTArrayCustom},
|
||||||
{NULL, NULL},
|
{NULL, NULL},
|
||||||
};
|
};
|
||||||
|
@ -45,6 +45,16 @@ enum SortOrder
|
|||||||
Sort_Descending = 1, /**< Descending order */
|
Sort_Descending = 1, /**< Descending order */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Data types for ADT Array Sorts
|
||||||
|
*/
|
||||||
|
enum SortType
|
||||||
|
{
|
||||||
|
Sort_Integer = 0,
|
||||||
|
Sort_Float,
|
||||||
|
Sort_String,
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sorts an array of integers.
|
* Sorts an array of integers.
|
||||||
*
|
*
|
||||||
@ -128,3 +138,38 @@ funcenum SortFunc2D
|
|||||||
* @noreturn
|
* @noreturn
|
||||||
*/
|
*/
|
||||||
native SortCustom2D(array[][], array_size, SortFunc2D:sortfunc, Handle:hndl=INVALID_HANDLE);
|
native SortCustom2D(array[][], array_size, SortFunc2D:sortfunc, Handle:hndl=INVALID_HANDLE);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sort an ADT Array. Specify the type as Integer, Float, or String.
|
||||||
|
*
|
||||||
|
* @param array Array Handle to sort
|
||||||
|
* @param order Sort order to use, same as other sorts.
|
||||||
|
* @param type Data type stored in the ADT Array
|
||||||
|
* @noreturn
|
||||||
|
*/
|
||||||
|
native SortADTArray(Handle:array, SortOrder:order = Sort_Ascending, SortType:type = Sort_Integer);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sort comparison function for ADT Array elements. Function provides you with
|
||||||
|
* indexes currently being sorted, use ADT Array functions to retrieve the
|
||||||
|
* index values and compare.
|
||||||
|
*
|
||||||
|
* @param index1 First index to compare.
|
||||||
|
* @param index2 Second index to compare.
|
||||||
|
* @param array Array that is being sorted (order is undefined).
|
||||||
|
* @param hndl Handle optionally passed in while sorting.
|
||||||
|
* @return -1 if first should go before second
|
||||||
|
* 0 if first is equal to second
|
||||||
|
* 1 if first should go after second
|
||||||
|
*/
|
||||||
|
functag SortFuncADTArray public(index1, index2, Handle:array, Handle:hndl);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Custom sorts an ADT Array. You must pass in a comparison function.
|
||||||
|
*
|
||||||
|
* @param array Array Handle to sort
|
||||||
|
* @param sortfunc Sort comparision function to use
|
||||||
|
* @param hndl Optional Handle to pass through the comparison calls.
|
||||||
|
* @noreturn
|
||||||
|
*/
|
||||||
|
native SortADTArrayCustom(Handle:array, SortFuncADTArray:sortfunc, Handle:hndl=INVALID_HANDLE);
|
@ -9,13 +9,17 @@ public Plugin:myinfo =
|
|||||||
url = "http://www.sourcemod.net/"
|
url = "http://www.sourcemod.net/"
|
||||||
};
|
};
|
||||||
|
|
||||||
public OnPluginStart(Handle:myself)
|
public OnPluginStart()
|
||||||
{
|
{
|
||||||
RegServerCmd("test_sort_ints", Command_TestSortInts)
|
RegServerCmd("test_sort_ints", Command_TestSortInts)
|
||||||
RegServerCmd("test_sort_floats", Command_TestSortFloats)
|
RegServerCmd("test_sort_floats", Command_TestSortFloats)
|
||||||
RegServerCmd("test_sort_strings", Command_TestSortStrings)
|
RegServerCmd("test_sort_strings", Command_TestSortStrings)
|
||||||
RegServerCmd("test_sort_1d", Command_TestSort1D)
|
RegServerCmd("test_sort_1d", Command_TestSort1D)
|
||||||
RegServerCmd("test_sort_2d", Command_TestSort2D)
|
RegServerCmd("test_sort_2d", Command_TestSort2D)
|
||||||
|
RegServerCmd("test_adtsort_ints", Command_TestSortADTInts)
|
||||||
|
RegServerCmd("test_adtsort_floats", Command_TestSortADTFloats)
|
||||||
|
RegServerCmd("test_adtsort_strings", Command_TestSortADTStrings)
|
||||||
|
RegServerCmd("test_adtsort_custom", Command_TestSortADTCustom)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*****************
|
/*****************
|
||||||
@ -138,7 +142,7 @@ public Action:Command_TestSortStrings(args)
|
|||||||
|
|
||||||
public Custom2DSort(String:elem1[], String:elem2[], String:array[][], Handle:hndl)
|
public Custom2DSort(String:elem1[], String:elem2[], String:array[][], Handle:hndl)
|
||||||
{
|
{
|
||||||
return StrCompare(elem1, elem2)
|
return strcmp(elem1, elem2)
|
||||||
}
|
}
|
||||||
|
|
||||||
public Action:Command_TestSort2D(args)
|
public Action:Command_TestSort2D(args)
|
||||||
@ -162,3 +166,142 @@ public Action:Command_TestSort2D(args)
|
|||||||
|
|
||||||
return Plugin_Handled
|
return Plugin_Handled
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*******************
|
||||||
|
* ADT ARRAY TESTS *
|
||||||
|
*******************/
|
||||||
|
// Int and floats work the same as normal comparisions. Strings are direct
|
||||||
|
// comparisions with no hacky memory stuff like Pawn arrays.
|
||||||
|
|
||||||
|
PrintADTArrayIntegers(Handle:array)
|
||||||
|
{
|
||||||
|
new size = GetArraySize(array);
|
||||||
|
for (new i=0; i<size;i++)
|
||||||
|
{
|
||||||
|
PrintToServer("array[%d] = %d", i, GetArrayCell(array, i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Action:Command_TestSortADTInts(args)
|
||||||
|
{
|
||||||
|
new Handle:array = CreateArray();
|
||||||
|
PushArrayCell(array, 6);
|
||||||
|
PushArrayCell(array, 7);
|
||||||
|
PushArrayCell(array, 3);
|
||||||
|
PushArrayCell(array, 2);
|
||||||
|
PushArrayCell(array, 8);
|
||||||
|
PushArrayCell(array, 5);
|
||||||
|
PushArrayCell(array, 0);
|
||||||
|
PushArrayCell(array, 1);
|
||||||
|
PushArrayCell(array, 4);
|
||||||
|
PushArrayCell(array, 9);
|
||||||
|
|
||||||
|
PrintToServer("Testing ascending sort:")
|
||||||
|
SortADTArray(array, Sort_Ascending, Sort_Integer)
|
||||||
|
PrintADTArrayIntegers(array)
|
||||||
|
|
||||||
|
PrintToServer("Testing descending sort:")
|
||||||
|
SortADTArray(array, Sort_Descending, Sort_Integer)
|
||||||
|
PrintADTArrayIntegers(array)
|
||||||
|
|
||||||
|
return Plugin_Handled
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintADTArrayFloats(Handle:array)
|
||||||
|
{
|
||||||
|
new size = GetArraySize(array);
|
||||||
|
for (new i=0; i<size;i++)
|
||||||
|
{
|
||||||
|
PrintToServer("array[%d] = %f", i, float:GetArrayCell(array, i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Action:Command_TestSortADTFloats(args)
|
||||||
|
{
|
||||||
|
new Handle:array = CreateArray();
|
||||||
|
PushArrayCell(array, 6.0);
|
||||||
|
PushArrayCell(array, 7.0);
|
||||||
|
PushArrayCell(array, 3.0);
|
||||||
|
PushArrayCell(array, 2.0);
|
||||||
|
PushArrayCell(array, 8.0);
|
||||||
|
PushArrayCell(array, 5.0);
|
||||||
|
PushArrayCell(array, 0.0);
|
||||||
|
PushArrayCell(array, 1.0);
|
||||||
|
PushArrayCell(array, 4.0);
|
||||||
|
PushArrayCell(array, 9.0);
|
||||||
|
|
||||||
|
PrintToServer("Testing ascending sort:")
|
||||||
|
SortADTArray(array, Sort_Ascending, Sort_Float)
|
||||||
|
PrintADTArrayFloats(array)
|
||||||
|
|
||||||
|
PrintToServer("Testing descending sort:")
|
||||||
|
SortADTArray(array, Sort_Descending, Sort_Float)
|
||||||
|
PrintADTArrayFloats(array)
|
||||||
|
|
||||||
|
return Plugin_Handled
|
||||||
|
}
|
||||||
|
|
||||||
|
PrintADTArrayStrings(Handle:array)
|
||||||
|
{
|
||||||
|
new size = GetArraySize(array);
|
||||||
|
decl String:buffer[64];
|
||||||
|
for (new i=0; i<size;i++)
|
||||||
|
{
|
||||||
|
GetArrayString(array, i, buffer, sizeof(buffer));
|
||||||
|
PrintToServer("array[%d] = %s", i, buffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Action:Command_TestSortADTStrings(args)
|
||||||
|
{
|
||||||
|
new Handle:array = CreateArray(ByteCountToCells(64));
|
||||||
|
PushArrayString(array, "faluco");
|
||||||
|
PushArrayString(array, "bailopan");
|
||||||
|
PushArrayString(array, "pm onoto");
|
||||||
|
PushArrayString(array, "damaged soul");
|
||||||
|
PushArrayString(array, "sniperbeamer");
|
||||||
|
PushArrayString(array, "sidluke");
|
||||||
|
PushArrayString(array, "johnny got his gun");
|
||||||
|
PushArrayString(array, "gabe newell");
|
||||||
|
PushArrayString(array, "hello");
|
||||||
|
PushArrayString(array, "WHAT?!");
|
||||||
|
|
||||||
|
PrintToServer("Testing ascending sort:")
|
||||||
|
SortADTArray(array, Sort_Ascending, Sort_String)
|
||||||
|
PrintADTArrayStrings(array)
|
||||||
|
|
||||||
|
PrintToServer("Testing descending sort:")
|
||||||
|
SortADTArray(array, Sort_Descending, Sort_String)
|
||||||
|
PrintADTArrayStrings(array)
|
||||||
|
|
||||||
|
return Plugin_Handled
|
||||||
|
}
|
||||||
|
|
||||||
|
public ArrayADTCustomCallback(index1, index2, Handle:array, Handle:hndl)
|
||||||
|
{
|
||||||
|
decl String:buffer1[64], String:buffer2[64];
|
||||||
|
GetArrayString(array, index1, buffer1, sizeof(buffer1));
|
||||||
|
GetArrayString(array, index2, buffer2, sizeof(buffer2));
|
||||||
|
|
||||||
|
return strcmp(buffer1, buffer2);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Action:Command_TestSortADTCustom(args)
|
||||||
|
{
|
||||||
|
new Handle:array = CreateArray(ByteCountToCells(64));
|
||||||
|
PushArrayString(array, "faluco");
|
||||||
|
PushArrayString(array, "bailopan");
|
||||||
|
PushArrayString(array, "pm onoto");
|
||||||
|
PushArrayString(array, "damaged soul");
|
||||||
|
PushArrayString(array, "sniperbeamer");
|
||||||
|
PushArrayString(array, "sidluke");
|
||||||
|
PushArrayString(array, "johnny got his gun");
|
||||||
|
PushArrayString(array, "gabe newell");
|
||||||
|
PushArrayString(array, "hello");
|
||||||
|
PushArrayString(array, "WHAT?!");
|
||||||
|
|
||||||
|
PrintToServer("Testing custom sort:")
|
||||||
|
SortADTArrayCustom(array, ArrayADTCustomCallback)
|
||||||
|
PrintADTArrayStrings(array);
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user