/* ** $Id: lbitlib.c,v 1.5 2010/07/02 11:38:13 roberto Exp roberto $ ** Standard library for bitwise operations ** See Copyright Notice in lua.h */ #define lbitlib_c #define LUA_LIB #include "lua.h" #include "lauxlib.h" #include "lualib.h" /* number of bits considered when shifting/rotating (must be a power of 2) */ #define NBITS 32 typedef LUA_INT32 b_int; typedef unsigned LUA_INT32 b_uint; static b_uint getuintarg (lua_State *L, int arg) { b_uint r; int isnum; lua_Number x = lua_tonumberx(L, arg, &isnum); if (!isnum) luaL_typeerror(L, arg, "number"); lua_number2uint(r, x); return r; } static b_uint andaux (lua_State *L) { int i, n = lua_gettop(L); b_uint r = ~(b_uint)0; for (i = 1; i <= n; i++) r &= getuintarg(L, i); return r; } static int b_and (lua_State *L) { b_uint r = andaux(L); lua_pushnumber(L, lua_uint2number(r)); return 1; } static int b_test (lua_State *L) { b_uint r = andaux(L); lua_pushboolean(L, r != 0); return 1; } static int b_or (lua_State *L) { int i, n = lua_gettop(L); b_uint r = 0; for (i = 1; i <= n; i++) r |= getuintarg(L, i); lua_pushnumber(L, lua_uint2number(r)); return 1; } static int b_xor (lua_State *L) { int i, n = lua_gettop(L); b_uint r = 0; for (i = 1; i <= n; i++) r ^= getuintarg(L, i); lua_pushnumber(L, lua_uint2number(r)); return 1; } static int b_not (lua_State *L) { b_uint r = ~getuintarg(L, 1); lua_pushnumber(L, lua_uint2number(r)); return 1; } static int b_shift (lua_State *L, int i) { b_uint r = getuintarg(L, 1); if (i < 0) { /* shift right? */ i = -i; if (i >= NBITS) r = 0; else r >>= i; } else { /* shift left */ if (i >= NBITS) r = 0; else r <<= i; } lua_pushnumber(L, lua_uint2number(r)); return 1; } static int b_lshift (lua_State *L) { return b_shift(L, luaL_checkint(L, 2)); } static int b_rshift (lua_State *L) { return b_shift(L, -luaL_checkint(L, 2)); } static int b_rot (lua_State *L, int i) { b_uint r = getuintarg(L, 1); i &= (NBITS - 1); /* i = i % NBITS */ r = (r << i) | (r >> (NBITS - i)); lua_pushnumber(L, lua_uint2number(r)); return 1; } static int b_rol (lua_State *L) { return b_rot(L, luaL_checkint(L, 2)); } static int b_ror (lua_State *L) { return b_rot(L, -luaL_checkint(L, 2)); } static const luaL_Reg bitlib[] = { {"band", b_and}, {"btest", b_test}, {"bor", b_or}, {"bxor", b_xor}, {"bnot", b_not}, {"lshift", b_lshift}, {"rshift", b_rshift}, {"rol", b_rol}, {"ror", b_ror}, {NULL, NULL} }; LUAMOD_API int luaopen_bit (lua_State *L) { luaL_newlib(L, bitlib); return 1; }