Merge pull request #188 from alliedmodders/tr-stack
Port adt_stack to transitional syntax.
This commit is contained in:
		
						commit
						758a7c955c
					
				@ -1,8 +1,8 @@
 | 
			
		||||
/**
 | 
			
		||||
* vim: set ts=4 :
 | 
			
		||||
* vim: set ts=4 sw=4 tw=99 noet :
 | 
			
		||||
* =============================================================================
 | 
			
		||||
* SourceMod
 | 
			
		||||
* Copyright (C) 2004-2008 AlliedModders LLC.  All rights reserved.
 | 
			
		||||
* Copyright (C) 2004-2014 AlliedModders LLC.  All rights reserved.
 | 
			
		||||
* =============================================================================
 | 
			
		||||
*
 | 
			
		||||
* This program is free software; you can redistribute it and/or modify it under
 | 
			
		||||
@ -32,6 +32,7 @@
 | 
			
		||||
#include <stdlib.h>
 | 
			
		||||
#include "common_logic.h"
 | 
			
		||||
#include "CellArray.h"
 | 
			
		||||
#include "handle_helpers.h"
 | 
			
		||||
 | 
			
		||||
HandleType_t htCellStack;
 | 
			
		||||
 | 
			
		||||
@ -293,6 +294,79 @@ static cell_t IsStackEmpty(IPluginContext *pContext, const cell_t *params)
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static cell_t ArrayStack_Pop(IPluginContext *pContext, const cell_t *params)
 | 
			
		||||
{
 | 
			
		||||
	OpenHandle<CellArray> array(pContext, params[1], htCellStack); 
 | 
			
		||||
	if (!array.Ok())
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	if (array->size() == 0)
 | 
			
		||||
		return pContext->ThrowNativeError("stack is empty");
 | 
			
		||||
 | 
			
		||||
	cell_t *blk = array->at(array->size() - 1);
 | 
			
		||||
	cell_t idx = (size_t)params[2];
 | 
			
		||||
 | 
			
		||||
	cell_t rval;
 | 
			
		||||
	if (params[3] == 0) {
 | 
			
		||||
		if (idx >= array->blocksize())
 | 
			
		||||
			return pContext->ThrowNativeError("Invalid block %d (blocksize: %d)", idx, array->blocksize());
 | 
			
		||||
		rval = blk[idx];
 | 
			
		||||
	} else {
 | 
			
		||||
		if (idx >= array->blocksize() * 4)
 | 
			
		||||
			return pContext->ThrowNativeError("Invalid byte %d (blocksize: %d bytes)", idx, array->blocksize() * 4);
 | 
			
		||||
		rval = (cell_t)*((char *)blk + idx);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	array->remove(array->size() - 1);
 | 
			
		||||
	return rval;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static cell_t ArrayStack_PopString(IPluginContext *pContext, const cell_t *params)
 | 
			
		||||
{
 | 
			
		||||
	OpenHandle<CellArray> array(pContext, params[1], htCellStack); 
 | 
			
		||||
	if (!array.Ok())
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	if (array->size() == 0)
 | 
			
		||||
		return pContext->ThrowNativeError("stack is empty");
 | 
			
		||||
 | 
			
		||||
	size_t idx = array->size() - 1;
 | 
			
		||||
	cell_t *blk = array->at(idx);
 | 
			
		||||
 | 
			
		||||
	cell_t *pWritten;
 | 
			
		||||
	pContext->LocalToPhysAddr(params[4], &pWritten);
 | 
			
		||||
 | 
			
		||||
	size_t numWritten;
 | 
			
		||||
	pContext->StringToLocalUTF8(params[2], params[3], (char *)blk, &numWritten);
 | 
			
		||||
	*pWritten = (cell_t)numWritten;
 | 
			
		||||
	array->remove(idx);
 | 
			
		||||
	return 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static cell_t ArrayStack_PopArray(IPluginContext *pContext, const cell_t *params)
 | 
			
		||||
{
 | 
			
		||||
	OpenHandle<CellArray> array(pContext, params[1], htCellStack); 
 | 
			
		||||
	if (!array.Ok())
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	if (array->size() == 0)
 | 
			
		||||
		return pContext->ThrowNativeError("stack is empty");
 | 
			
		||||
 | 
			
		||||
	cell_t *addr;
 | 
			
		||||
	pContext->LocalToPhysAddr(params[2], &addr);
 | 
			
		||||
 | 
			
		||||
	size_t idx = array->size() - 1;
 | 
			
		||||
	cell_t *blk = array->at(idx);
 | 
			
		||||
	size_t indexes = array->blocksize();
 | 
			
		||||
 | 
			
		||||
	if (params[3] != -1 && (size_t)params[3] <= array->blocksize())
 | 
			
		||||
		indexes = params[3];
 | 
			
		||||
 | 
			
		||||
	memcpy(addr, blk, sizeof(cell_t) * indexes);
 | 
			
		||||
	array->remove(idx);
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
REGISTER_NATIVES(cellStackNatives)
 | 
			
		||||
{
 | 
			
		||||
	{"CreateStack",					CreateStack},
 | 
			
		||||
@ -303,5 +377,16 @@ REGISTER_NATIVES(cellStackNatives)
 | 
			
		||||
	{"PushStackArray",				PushStackArray},
 | 
			
		||||
	{"PushStackCell",				PushStackCell},
 | 
			
		||||
	{"PushStackString",				PushStackString},
 | 
			
		||||
 | 
			
		||||
	// Transitional syntax support.
 | 
			
		||||
	{"ArrayStack.ArrayStack",		CreateStack},
 | 
			
		||||
	{"ArrayStack.Pop",				ArrayStack_Pop},
 | 
			
		||||
	{"ArrayStack.PopString",		ArrayStack_PopString},
 | 
			
		||||
	{"ArrayStack.PopArray",			ArrayStack_PopArray},
 | 
			
		||||
	{"ArrayStack.Push",				PushStackCell},
 | 
			
		||||
	{"ArrayStack.PushString",		PushStackString},
 | 
			
		||||
	{"ArrayStack.PushArray",		PushStackArray},
 | 
			
		||||
	{"ArrayStack.Empty.get",		IsStackEmpty},
 | 
			
		||||
 | 
			
		||||
	{NULL,							NULL},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
@ -1,7 +1,7 @@
 | 
			
		||||
/**
 | 
			
		||||
 * vim: set ts=4 :
 | 
			
		||||
 * vim: set ts=4 sw=4 tw=99 noet :
 | 
			
		||||
 * =============================================================================
 | 
			
		||||
 * SourceMod (C)2004-2008 AlliedModders LLC.  All rights reserved.
 | 
			
		||||
 * SourceMod (C)2004-2014 AlliedModders LLC.  All rights reserved.
 | 
			
		||||
 * =============================================================================
 | 
			
		||||
 *
 | 
			
		||||
 * This file is part of the SourceMod/SourcePawn SDK.
 | 
			
		||||
@ -35,6 +35,82 @@
 | 
			
		||||
#endif
 | 
			
		||||
#define _adt_stack_included
 | 
			
		||||
 | 
			
		||||
methodmap ArrayStack < Handle
 | 
			
		||||
{
 | 
			
		||||
	// Creates a stack structure.  A stack is a LIFO (last in, first out) 
 | 
			
		||||
	// vector (array) of items.  It has O(1) insertion and O(1) removal.
 | 
			
		||||
	//
 | 
			
		||||
	// Stacks have two operations: Push (adding an item) and Pop (removes 
 | 
			
		||||
	// items in reverse-push order).
 | 
			
		||||
	// 
 | 
			
		||||
	// The contents of the stack are uniform; i.e. storing a string and then 
 | 
			
		||||
	// retrieving it as an integer is NOT the same as StringToInt()!
 | 
			
		||||
	//
 | 
			
		||||
	// The "blocksize" determines how many cells each slot has; it cannot
 | 
			
		||||
	// be changed after creation.
 | 
			
		||||
	//
 | 
			
		||||
	// @param blocksize    The number of cells each entry in the stack can 
 | 
			
		||||
	//                     hold.  For example, 32 cells is equivalent to:
 | 
			
		||||
	//                     new Array[X][32]
 | 
			
		||||
	public native ArrayStack(int blocksize=1);
 | 
			
		||||
 | 
			
		||||
	// Pushes a value onto the end of the stack, adding a new index.
 | 
			
		||||
	//
 | 
			
		||||
	// This may safely be used even if the stack has a blocksize
 | 
			
		||||
	// greater than 1.
 | 
			
		||||
	//
 | 
			
		||||
	// @param value        Value to push.
 | 
			
		||||
	public native void Push(any value);
 | 
			
		||||
 | 
			
		||||
	// Pushes a copy of a string onto the end of a stack, truncating it if it
 | 
			
		||||
	// is too big.
 | 
			
		||||
	//
 | 
			
		||||
	// @param value        String to push.
 | 
			
		||||
	public native void PushString(const char[] value);
 | 
			
		||||
 | 
			
		||||
	// Pushes a copy of an array of cells onto the end of a stack. The cells
 | 
			
		||||
	// are pushed as a block (i.e. the entire array takes up one stack slot),
 | 
			
		||||
	// rather than pushing each cell individually.
 | 
			
		||||
	//
 | 
			
		||||
	// @param stack        Stack Handle.
 | 
			
		||||
	// @param values       Block of values to copy.
 | 
			
		||||
	// @param size         If not set, the number of elements copied from the array
 | 
			
		||||
	//                     will be equal to the blocksize.  If set higher than the 
 | 
			
		||||
	//                     blocksize, the operation will be truncated.
 | 
			
		||||
	public native void PushArray(const any[] values, int size=-1);
 | 
			
		||||
 | 
			
		||||
	// Pops a cell value from a stack.
 | 
			
		||||
	//
 | 
			
		||||
	// @param block        Optionally specify which block to read from
 | 
			
		||||
	//                     (useful if the blocksize > 0).
 | 
			
		||||
	// @param asChar       Optionally read as a byte instead of a cell.
 | 
			
		||||
	// @return             True on success, false if the stack is empty.
 | 
			
		||||
	// @error              The stack is empty.
 | 
			
		||||
	public native any Pop(int block=0, bool asChar=false);
 | 
			
		||||
 | 
			
		||||
	// Pops a string value from a stack.
 | 
			
		||||
	//
 | 
			
		||||
	// @param buffer       Buffer to store string.
 | 
			
		||||
	// @param maxlength    Maximum size of the buffer.
 | 
			
		||||
	// @oaram written      Number of characters written to buffer, not including
 | 
			
		||||
	//                     the null terminator.
 | 
			
		||||
	// @error              The stack is empty.
 | 
			
		||||
	public native void PopString(char[] buffer, int maxlength, int &written = 0);
 | 
			
		||||
 | 
			
		||||
	// Pops an array of cells from a stack.
 | 
			
		||||
	//
 | 
			
		||||
	// @param buffer       Buffer to store the array in.
 | 
			
		||||
	// @param size         If not set, assumes the buffer size is equal to the
 | 
			
		||||
	//                     blocksize.  Otherwise, the size passed is used.
 | 
			
		||||
	// @error              The stack is empty.
 | 
			
		||||
	public native void PopArray(any[] buffer, int size=-1);
 | 
			
		||||
 | 
			
		||||
	// Returns true if the stack is empty, false otherwise.
 | 
			
		||||
	property bool Empty {
 | 
			
		||||
		public native get();
 | 
			
		||||
	}
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Creates a stack structure.  A stack is a LIFO (last in, first out) 
 | 
			
		||||
 * vector (array) of items.  It has O(1) insertion and O(1) removal.
 | 
			
		||||
@ -53,7 +129,7 @@
 | 
			
		||||
 *						new Array[X][32]
 | 
			
		||||
 * @return				New stack Handle.
 | 
			
		||||
 */
 | 
			
		||||
native Handle:CreateStack(blocksize=1);
 | 
			
		||||
native ArrayStack CreateStack(int blocksize=1);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pushes a value onto the end of the stack, adding a new index.
 | 
			
		||||
@ -63,24 +139,22 @@ native Handle:CreateStack(blocksize=1);
 | 
			
		||||
 *
 | 
			
		||||
 * @param stack			Stack Handle.
 | 
			
		||||
 * @param value			Value to push.
 | 
			
		||||
 * @noreturn
 | 
			
		||||
 * @error				Invalid Handle or out of memory.
 | 
			
		||||
 */
 | 
			
		||||
native PushStackCell(Handle:stack, any:value);
 | 
			
		||||
native void PushStackCell(Handle stack, any value);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pushes a string onto the end of a stack, truncating it if it is 
 | 
			
		||||
 * Pushes a copy of a string onto the end of a stack, truncating it if it is 
 | 
			
		||||
 * too big.
 | 
			
		||||
 *
 | 
			
		||||
 * @param stack			Stack Handle.
 | 
			
		||||
 * @param value			String to push.
 | 
			
		||||
 * @noreturn
 | 
			
		||||
 * @error				Invalid Handle or out of memory.
 | 
			
		||||
 */
 | 
			
		||||
native PushStackString(Handle:stack, const String:value[]);
 | 
			
		||||
native void PushStackString(Handle stack, const char[] value);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pushes an array of cells onto the end of a stack.  The cells
 | 
			
		||||
 * Pushes a copy of an array of cells onto the end of a stack.  The cells
 | 
			
		||||
 * are pushed as a block (i.e. the entire array takes up one stack slot),
 | 
			
		||||
 * rather than pushing each cell individually.
 | 
			
		||||
 *
 | 
			
		||||
@ -89,10 +163,9 @@ native PushStackString(Handle:stack, const String:value[]);
 | 
			
		||||
 * @param size			If not set, the number of elements copied from the array
 | 
			
		||||
 *						will be equal to the blocksize.  If set higher than the 
 | 
			
		||||
 *						blocksize, the operation will be truncated.
 | 
			
		||||
 * @noreturn
 | 
			
		||||
 * @error				Invalid Handle or out of memory.
 | 
			
		||||
 */
 | 
			
		||||
native PushStackArray(Handle:stack, const any:values[], size=-1);
 | 
			
		||||
native void PushStackArray(Handle stack, const any[] values, int size=-1);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pops a cell value from a stack.
 | 
			
		||||
@ -105,7 +178,7 @@ native PushStackArray(Handle:stack, const any:values[], size=-1);
 | 
			
		||||
 * @return				True on success, false if the stack is empty.
 | 
			
		||||
 * @error				Invalid Handle.
 | 
			
		||||
 */
 | 
			
		||||
native bool:PopStackCell(Handle:stack, &any:value, block=0, bool:asChar=false);
 | 
			
		||||
native bool PopStackCell(Handle stack, any &value, int block=0, bool asChar=false);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pops a string value from a stack.
 | 
			
		||||
@ -116,7 +189,7 @@ native bool:PopStackCell(Handle:stack, &any:value, block=0, bool:asChar=false);
 | 
			
		||||
 * @return				True on success, false if the stack is empty.
 | 
			
		||||
 * @error				Invalid Handle.
 | 
			
		||||
 */
 | 
			
		||||
native bool:PopStackString(Handle:stack, String:buffer[], maxlength, &written=0);
 | 
			
		||||
native bool PopStackString(Handle stack, char[] buffer, int maxlength, int &written=0);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pops an array of cells from a stack.
 | 
			
		||||
@ -128,7 +201,7 @@ native bool:PopStackString(Handle:stack, String:buffer[], maxlength, &written=0)
 | 
			
		||||
 * @return				True on success, false if the stack is empty.
 | 
			
		||||
 * @error				Invalid Handle.
 | 
			
		||||
 */
 | 
			
		||||
native bool:PopStackArray(Handle:stack, any:buffer[], size=-1);
 | 
			
		||||
native bool PopStackArray(Handle stack, any[] buffer, int size=-1);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Checks if a stack is empty.
 | 
			
		||||
@ -137,7 +210,7 @@ native bool:PopStackArray(Handle:stack, any:buffer[], size=-1);
 | 
			
		||||
 * @return				True if empty, false if not empty.
 | 
			
		||||
 * @error				Invalid Handle.
 | 
			
		||||
 */
 | 
			
		||||
native bool:IsStackEmpty(Handle:stack);
 | 
			
		||||
native bool IsStackEmpty(Handle stack);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Pops a value off a stack, ignoring it completely.
 | 
			
		||||
@ -146,9 +219,8 @@ native bool:IsStackEmpty(Handle:stack);
 | 
			
		||||
 * @return				True if something was popped, false otherwise.
 | 
			
		||||
 * @error				Invalid Handle.
 | 
			
		||||
 */
 | 
			
		||||
stock PopStack(Handle:stack)
 | 
			
		||||
stock bool PopStack(Handle stack)
 | 
			
		||||
{
 | 
			
		||||
	new value;
 | 
			
		||||
	
 | 
			
		||||
	return PopStackCell(stack, value);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -16,38 +16,36 @@ public OnPluginStart()
 | 
			
		||||
 | 
			
		||||
public Action:Test_Stack(args)
 | 
			
		||||
{
 | 
			
		||||
	new Handle:stack;
 | 
			
		||||
	new test[20]
 | 
			
		||||
	decl String:buffer[42];
 | 
			
		||||
	int test[20];
 | 
			
		||||
	char buffer[42];
 | 
			
		||||
	
 | 
			
		||||
	test[0] = 5
 | 
			
		||||
	test[1] = 7
 | 
			
		||||
	
 | 
			
		||||
	stack = CreateStack(30);
 | 
			
		||||
	PushStackCell(stack, 50);
 | 
			
		||||
	PushStackArray(stack, test, 2);
 | 
			
		||||
	PushStackArray(stack, test, 2);
 | 
			
		||||
	PushStackString(stack, "space craaab");
 | 
			
		||||
	PushStackCell(stack, 12);
 | 
			
		||||
	ArrayStack stack = ArrayStack(30);
 | 
			
		||||
	stack.Push(50);
 | 
			
		||||
	stack.PushArray(test, 2);
 | 
			
		||||
	stack.PushArray(test, 2);
 | 
			
		||||
	stack.PushString("space craaab");
 | 
			
		||||
	stack.Push(12);
 | 
			
		||||
	
 | 
			
		||||
	PrintToServer("empty? %d", IsStackEmpty(stack));
 | 
			
		||||
	PrintToServer("empty? %d", stack.Empty);
 | 
			
		||||
	
 | 
			
		||||
	PopStack(stack);
 | 
			
		||||
	PopStackString(stack, buffer, sizeof(buffer));
 | 
			
		||||
	stack.Pop();
 | 
			
		||||
	stack.PopString(buffer, sizeof(buffer));
 | 
			
		||||
	PrintToServer("popped: \"%s\"", buffer);
 | 
			
		||||
	test[0] = 0
 | 
			
		||||
	test[1] = 0
 | 
			
		||||
	PrintToServer("values: %d, %d", test[0], test[1]);
 | 
			
		||||
	PopStackArray(stack, test, 2);
 | 
			
		||||
	stack.PopArray(test, 2);
 | 
			
		||||
	PrintToServer("popped: %d, %d", test[0], test[1]);
 | 
			
		||||
	PopStackCell(stack, test[0], 1);
 | 
			
		||||
	test[0] = stack.Pop(1);
 | 
			
		||||
	PrintToServer("popped: x, %d", test[0]);
 | 
			
		||||
	PopStackCell(stack, test[0]);
 | 
			
		||||
	test[0] = stack.Pop();
 | 
			
		||||
	PrintToServer("popped: %d", test[0]);
 | 
			
		||||
	
 | 
			
		||||
	PrintToServer("empty? %d", IsStackEmpty(stack));
 | 
			
		||||
	
 | 
			
		||||
	CloseHandle(stack);
 | 
			
		||||
	PrintToServer("empty? %d", stack.Empty);
 | 
			
		||||
	
 | 
			
		||||
	delete stack;
 | 
			
		||||
	return Plugin_Handled;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
		Loading…
	
		Reference in New Issue
	
	Block a user