// vim: set ts=4 sw=4 tw=99 noet: // // AMX Mod X, based on AMX Mod by Aleksander Naszko ("OLO"). // Copyright (C) The AMX Mod X Development Team. // // This software is licensed under the GNU General Public License, version 3 or higher. // Additional exceptions apply. For full license details, see LICENSE.txt or visit: // https://alliedmods.net/amxmodx-license #include "amxmodx.h" #include #include #include "datastructs.h" /*********************************** * About the double array hack * *************************** Double arrays in Pawn are vectors offset by the current offset. For example: new array[2][2] In this array, index 0 contains the offset from the current offset which results in the final vector [2] (at [0][2]). Meaning, to dereference [1][2], it is equivalent to: address = &array[1] + array[1] + 2 * sizeof(cell) The fact that each offset is from the _current_ position rather than the _base_ position is very important. It means that if you to try to swap vector positions, the offsets will no longer match, because their current position has changed. A simple and ingenious way around this is to back up the positions in a separate array, then to overwrite each position in the old array with absolute indices. Pseudo C++ code: cell *array; //assumed to be set to the 2+D array cell *old_offsets = new cell[2]; for (int i=0; i<2; i++) { old_offsets = array[i]; array[i] = i; } Now, you can swap the array indices with no problem, and do a reverse-lookup to find the original addresses. After sorting/modification is done, you must relocate the new indices. For example, if the two vectors in our demo array were swapped, array[0] would be 1 and array[1] would be 0. This is invalid to the virtual machine. Luckily, this is also simple -- all the information is there. for (int i=0; i<2; i++) { //get the # of the vector we want to relocate in cell vector_index = array[i]; //get the real address of this vector char *real_address = (char *)array + (vector_index * sizeof(cell)) + old_offsets[vector_index]; //calc and store the new distance offset array[i] = real_address - ( (char *)array + (vector_index + sizeof(cell)) ) } Note that the inner expression can be heavily reduced; it is expanded for readability. **********************************/ enum SortOrder { Sort_Ascending = 0, Sort_Descending = 1, Sort_Random = 2, }; int sort_ints_asc(const void *int1, const void *int2) { return (*(int *)int1) - (*(int *)int2); } int sort_ints_desc(const void *int1, const void *int2) { return (*(int *)int2) - (*(int *)int1); } void sort_random(cell *array, cell size) { srand((unsigned int)time(NULL)); for (int i = size-1; i > 0; i--) { int n = rand() % (i + 1); if (array[i] != array[n]) { array[i] ^= array[n]; array[n] ^= array[i]; array[i] ^= array[n]; } } } static cell AMX_NATIVE_CALL SortIntegers(AMX *amx, cell *params) { cell *array = get_amxaddr(amx, params[1]); cell array_size = params[2]; cell type = params[3]; if (type == Sort_Ascending) { qsort(array, array_size, sizeof(cell), sort_ints_asc); } else if (type == Sort_Descending) { qsort(array, array_size, sizeof(cell), sort_ints_desc); } else { sort_random(array, array_size); } return 1; } int sort_floats_asc(const void *float1, const void *float2) { REAL r1 = *(REAL *)float1; REAL r2 = *(REAL *)float2; if (r1 < r2) { return -1; } else if (r2 < r1) { return 1; } else { return 0; } } int sort_floats_desc(const void *float1, const void *float2) { REAL r1 = *(REAL *)float1; REAL r2 = *(REAL *)float2; if (r1 < r2) { return 1; } else if (r2 < r1) { return -1; } else { return 0; } } static cell AMX_NATIVE_CALL SortFloats(AMX *amx, cell *params) { cell *array = get_amxaddr(amx, params[1]); cell array_size = params[2]; cell type = params[3]; if (type == Sort_Ascending) { qsort(array, array_size, sizeof(cell), sort_floats_asc); } else if (type == Sort_Descending) { qsort(array, array_size, sizeof(cell), sort_floats_desc); } else { sort_random(array, array_size); } return 1; } static cell *g_CurStringArray = NULL; static cell *g_CurRebaseMap = NULL; int sort_strings_asc(const void *blk1, const void *blk2) { cell reloc1 = *(cell *)blk1; cell reloc2 = *(cell *)blk2; register cell *str1 = (cell *)((char *)(&g_CurStringArray[reloc1]) + g_CurRebaseMap[reloc1]); register cell *str2 = (cell *)((char *)(&g_CurStringArray[reloc2]) + g_CurRebaseMap[reloc2]); while (*str1 == *str2++) { if (*str1++ == 0) { return 0; } } return (*str1 - *(str2 - 1)); } int sort_strings_desc(const void *blk1, const void *blk2) { cell reloc1 = *(cell *)blk1; cell reloc2 = *(cell *)blk2; register cell *str1 = (cell *)((char *)(&g_CurStringArray[reloc1]) + g_CurRebaseMap[reloc1]); register cell *str2 = (cell *)((char *)(&g_CurStringArray[reloc2]) + g_CurRebaseMap[reloc2]); while (*str1 == *str2++) { if (*str1++ == 0) { return 0; } } return (*(str2 - 1) - *str1); } static cell AMX_NATIVE_CALL SortStrings(AMX *amx, cell *params) { cell *array = get_amxaddr(amx, params[1]); cell array_size = params[2]; cell type = params[3]; /** HACKHACK - back up the old indices, replace the indices with something easier */ cell amx_addr, *phys_addr; int err; if ((err=amx_Allot(amx, array_size, &amx_addr, &phys_addr)) != AMX_ERR_NONE) { LogError(amx, err, "Ran out of memory"); return 0; } g_CurStringArray = array; g_CurRebaseMap = phys_addr; for (int i=0; i g_AMXSortStack; int sort1d_amx_custom(const void *elem1, const void *elem2) { cell c1 = *(cell *)elem1; cell c2 = *(cell *)elem2; sort_info *pInfo = g_AMXSortStack.front(); return executeForwards(pInfo->pfn, c1, c2, pInfo->array_addr, pInfo->data_addr, pInfo->data_size); } static cell AMX_NATIVE_CALL SortCustom1D(AMX *amx, cell *params) { cell *array = get_amxaddr(amx, params[1]); cell array_size = params[2]; int len; const char *funcname = get_amxstring(amx, params[3], 0, len); int pfn = registerSPForwardByName(amx, funcname, FP_CELL, FP_CELL, FP_CELL, FP_CELL, FP_CELL, FP_DONE); if (pfn < 0) { LogError(amx, AMX_ERR_NATIVE, "The public function \"%s\" was not found.", funcname); return 0; } sort_info *pInfo = new sort_info; pInfo->pfn = pfn; pInfo->data_addr = params[4]; pInfo->data_size = params[5]; pInfo->array_addr = params[1]; pInfo->array_remap = NULL; pInfo->array_base = NULL; g_AMXSortStack.push(pInfo); qsort(array, array_size, sizeof(cell), sort1d_amx_custom); g_AMXSortStack.pop(); unregisterSPForward(pfn); delete pInfo; return 1; } int sort2d_amx_custom(const void *elem1, const void *elem2) { cell c1 = *(cell *)elem1; cell c2 = *(cell *)elem2; sort_info *pInfo = g_AMXSortStack.front(); cell c1_addr = pInfo->array_addr + (c1 * sizeof(cell)) + pInfo->array_remap[c1]; cell c2_addr = pInfo->array_addr + (c2 * sizeof(cell)) + pInfo->array_remap[c2]; //cell *c1_r = get_amxaddr(pInfo->amx, c1_addr); //cell *c2_r = get_amxaddr(pInfo->amx, c2_addr); return executeForwards(pInfo->pfn, c1_addr, c2_addr, pInfo->array_addr, pInfo->data_addr, pInfo->data_size); } static cell AMX_NATIVE_CALL SortCustom2D(AMX *amx, cell *params) { cell *array = get_amxaddr(amx, params[1]); cell array_size = params[2]; int len; const char *funcname = get_amxstring(amx, params[3], 0, len); /** back up the old indices, replace the indices with something easier */ cell amx_addr, *phys_addr; int err; if ((err=amx_Allot(amx, array_size, &amx_addr, &phys_addr)) != AMX_ERR_NONE) { LogError(amx, err, "Ran out of memory"); return 0; } int pfn = registerSPForwardByName(amx, funcname, FP_CELL, FP_CELL, FP_CELL, FP_CELL, FP_CELL, FP_DONE); if (pfn < 0) { amx_Release(amx, amx_addr); LogError(amx, AMX_ERR_NATIVE, "The public function \"%s\" was not found.", funcname); return 0; } sort_info *pInfo = new sort_info; pInfo->pfn = pfn; pInfo->data_addr = params[4]; pInfo->data_size = params[5]; pInfo->array_addr = params[1]; pInfo->amx = amx; /** Same process as in strings, back up the old indices for later fixup */ pInfo->array_base = array; pInfo->array_remap = phys_addr; for (int i=0; ipfn); delete pInfo; 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); } void sort_adt_random(CellArray *cArray) { size_t arraysize = cArray->size(); srand((unsigned int)time(NULL)); for (int i = arraysize-1; i > 0; i--) { int n = rand() % (i + 1); cArray->swap(i, n); } } static cell AMX_NATIVE_CALL SortADTArray(AMX *amx, cell *params) { CellArray* vec = HandleToVector(amx, params[1]); if (vec == NULL) { return 0; } cell order = params[2]; if (order == Sort_Random) { sort_adt_random(vec); return 1; } cell type = params[3]; size_t arraysize = vec->size(); size_t blocksize = vec->blocksize(); cell *array = vec->base(); if (type == Sort_Integer) { if (order == Sort_Ascending) { qsort(array, arraysize, blocksize * sizeof(cell), sort_ints_asc); } else { qsort(array, arraysize, blocksize * sizeof(cell), sort_ints_desc); } } else if (type == Sort_Float) { if (order == Sort_Ascending) { qsort(array, arraysize, blocksize * sizeof(cell), sort_floats_asc); } else { qsort(array, arraysize, blocksize * sizeof(cell), sort_floats_desc); } } else if (type == Sort_String) { if (order == Sort_Ascending) { qsort(array, arraysize, blocksize * sizeof(cell), sort_adtarray_strings_asc); } else { qsort(array, arraysize, blocksize * sizeof(cell), sort_adtarray_strings_desc); } } return 1; } AMX_NATIVE_INFO g_SortNatives[] = { {"SortIntegers", SortIntegers}, {"SortFloats", SortFloats}, {"SortStrings", SortStrings}, {"SortCustom1D", SortCustom1D}, {"SortCustom2D", SortCustom2D}, {"SortADTArray", SortADTArray}, {NULL, NULL}, };