如何将luajit指针转换为字符串再转回来?

我需要些帮助将luajit指针转换为字符串再转回来。

首先我定义了这个 ctype:

ffi.cdef[[
    typedef struct {
        unsigned char Bytes[16];
    } EncryptionKeys[100000000];

void* malloc(size_t);
void free(void*);
]]

然后使用 malloc 分配了一些内存并创建了“EncryptionKeys”变量。

local EncryptionKeyMemoryAddress = ffi.C.malloc(ffi.sizeof("EncryptionKeys"))

local EncryptionKeys = ffi.cast("EncryptionKeys(&)", EncryptionKeyMemoryAddress)

我首先使用以下方法将变量转换为 lua 字符串:

ffi.string(EncryptionKeyMemoryAddress)

但我无法弄清如何将其转换回来! 有人可以帮帮我吗?

FYI:我正在将 'EncryptionKeyMemoryAddress' 变量传递给 lua lane 的一个函数参数(https://lualanes.github.io/lanes/)。

编辑: 以下是我正在工作的代码部分: 这是我的服务器的客户端管理模块,管理着一组所有可以访问服务器上连接的任何客户端的 lua 状态的列表。他们都使用共享内存部分,我希望他们可以使用指针访问它。


local ClientFFIString = [[

    typedef struct {
        unsigned char Bytes[16];
    } EncryptionKeys[100000000];

    void* malloc(size_t);
    void free(void*);
]]

ffi.cdef(Matchpools.FFIString)

local EncryptionKeyMemoryAddress = ffi.C.malloc(ffi.sizeof("EncryptionKeys"))

--------------------------------------------
function ClientManagers.CreateNewClientManager()

    local EncryptionKeys = ffi.cast("EncryptionKeys(&)", EncryptionKeyMemoryAddress)

    EncryptionKeys[0].Bytes[0] = 24

    print("___a", EncryptionKeys[0].Bytes[0])

    local NewIndex = #ClientManagers.List+1
    ClientManagers.List[NewIndex] = ClientManagerFunc(
        ClientFFIString,
        ffi.string(EncryptionKeysMemoryAddress)
    )

end

--------------------------------------------
local ClientManagerFunc = Lanes.gen("*", function(ClientFFIString, EncryptionKeysMemoryAddress)

    ffi = require("ffi")

    ffi.cdef(ClientFFIString)

    local EncryptionKeys = ffi.cast("EncryptionKeys(&)", EncryptionKeyMemoryAddress)

    print("___a", EncryptionKeys[0].Bytes[0])
    -- 我希望这也像创建此 lua 状态的函数一样是 24

    local ClientManagerRunning = true
    while ClientManagerRunning do

        --local dt = GetDt()

        --UpdateClientData(dt)

        --UpdateMatchmaking(dt)

    end

end)
点赞
用户1847592
用户1847592

你可以将 Lua 字符串转换为结构体指针(并且之后可以像数组一样使用):

ffi.cdef"typedef struct {unsigned char Bytes[16];} Key;"
ptr=ffi.cast("Key *", your_string)
print("First Byte of the First Key in your Array:", ptr[0].Bytes[0])

更新:

让我们测试一下它如何使用包含三个键的数组:

local ffi = require'ffi'
ffi.cdef"typedef struct {unsigned char Bytes[16];} Key;"
local your_string = string.char(11):rep(16)..string.char(22):rep(16)..string.char(33):rep(16)
local ptr=ffi.cast("Key *", your_string)
print("First Byte of the First Key in your Array:", ptr[0].Bytes[0])
print("First Byte of the Second Key in your Array:", ptr[1].Bytes[0])
print("First Byte of the Third Key in your Array:", ptr[2].Bytes[0])

它会打印出 11,22,33。


更新 2

传递缓冲区的地址而不是缓冲区的内容

在主线程中

-- allocate the buffer
local Keys = ffi.cast("EncryptionKeys&", ffi.C.malloc(ffi.sizeof("EncryptionKeys")))
-- write to the buffer
Keys[2].Bytes[5] = 42
-- create string containing 64-bit address
local string_to_send = tostring(ffi.cast("uint64_t", Keys))
-- Send string_to_send to a lane

在子线程中

-- receive the string
local received_string = .....
-- restore the buffer pointer
local Keys = ffi.cast("EncryptionKeys&", loadstring("return "..received_string)())
-- read the data from the buffer
print(Keys[2].Bytes[5])
2021-04-16 03:29:48