/* * wslua_frame_info.c * * Wireshark's interface to the Lua Programming Language * for frame data and meta-data from a capture file. * * (c) 2014, Hadriel Kaplan * * Wireshark - Network traffic analyzer * By Gerald Combs * Copyright 1998 Gerald Combs * * SPDX-License-Identifier: GPL-2.0-or-later */ #include "config.h" #define WS_LOG_DOMAIN LOG_DOMAIN_WSLUA #include "wslua_file_common.h" #include /* WSLUA_CONTINUE_MODULE File */ WSLUA_CLASS_DEFINE(FrameInfo,FAIL_ON_NULL_OR_EXPIRED("FrameInfo")); /* A FrameInfo object, passed into Lua as an argument by FileHandler callback functions (e.g., `read`, `seek_read`, etc.). This object represents frame data and meta-data (data about the frame/packet) for a given `read`/`seek_read`/`write`++'++s frame. This object's fields are written-to/set when used by read function callbacks, and read-from/get when used by file write function callbacks. In other words, when the Lua plugin's FileHandler `read`/`seek_read`/etc. functions are invoked, a FrameInfo object will be passed in as one of the arguments, and its fields should be written-to/set based on the frame information read from the file; whereas when the Lua plugin's `FileHandler.write()` function is invoked, the `FrameInfo` object passed in should have its fields read-from/get, to write that frame information to the file. */ FrameInfo* push_FrameInfo(lua_State* L, wtap_rec *rec, Buffer* buf) { FrameInfo f = (FrameInfo) g_malloc0(sizeof(struct _wslua_phdr)); f->rec = rec; f->buf = buf; f->expired = false; return pushFrameInfo(L,f); } WSLUA_METAMETHOD FrameInfo__tostring(lua_State* L) { /* Generates a string of debug info for the FrameInfo */ FrameInfo fi = toFrameInfo(L,1); if (!fi) { lua_pushstring(L,"FrameInfo pointer is NULL!"); } else { if (fi->rec) lua_pushfstring(L, "FrameInfo: rec_type=%u, presence_flags=%d, caplen=%d, len=%d, pkt_encap=%d, block='%p'", fi->rec->rec_type, fi->rec->presence_flags, fi->rec->rec_header.packet_header.caplen, fi->rec->rec_header.packet_header.len, fi->rec->rec_header.packet_header.pkt_encap, fi->rec->block); else lua_pushstring(L, "FrameInfo rec pointer is NULL!"); } WSLUA_RETURN(1); /* String of debug information. */ } /* XXX: should this function be a method of File instead? */ WSLUA_METHOD FrameInfo_read_data(lua_State* L) { /* Tells Wireshark to read directly from given file into frame data buffer, for length bytes. Returns true if succeeded, else false. */ #define WSLUA_ARG_FrameInfo_read_data_FILE 2 /* The File object userdata, provided by Wireshark previously in a reading-based callback. */ #define WSLUA_ARG_FrameInfo_read_data_LENGTH 3 /* The number of bytes to read from the file at the current cursor position. */ FrameInfo fi = checkFrameInfo(L,1); File fh = checkFile(L,WSLUA_ARG_FrameInfo_read_data_FILE); uint32_t len = wslua_checkuint32(L, WSLUA_ARG_FrameInfo_read_data_LENGTH); int err = 0; char *err_info = NULL; if (!fi->buf || !fh->file) { luaL_error(L, "FrameInfo read_data() got null buffer or file pointer internally"); return 0; } if (!wtap_read_packet_bytes(fh->file, fi->buf, len, &err, &err_info)) { lua_pushboolean(L, false); if (err_info) { lua_pushstring(L, err_info); g_free(err_info); /* is this right? */ } else lua_pushnil(L); lua_pushinteger(L, err); return 3; } lua_pushboolean(L, true); WSLUA_RETURN(1); /* True if succeeded, else returns false along with the error number and string error description. */ } /* free the struct we created, but not the rec/buf it points to */ static int FrameInfo__gc(lua_State* L) { FrameInfo fi = toFrameInfo(L,1); g_free(fi); return 0; } /* WSLUA_ATTRIBUTE FrameInfo_comment RW Table of comments in this frame. */ static int FrameInfo_get_comment (lua_State* L) { FrameInfo fi = checkFrameInfo(L,1); #define FRAMEINFO_COMMENTS_TABLE 2 char *comment = NULL; wtap_block_t block = NULL; unsigned i = 0; unsigned n_comments = 0; block = fi->rec->block; // XXX - how to get the user-edited block, if any? n_comments = wtap_block_count_option(block, OPT_COMMENT); lua_createtable(L, n_comments, 0); for (i = 0; i < n_comments; i++) { comment = NULL; lua_pushinteger(L, i+1); if (WTAP_OPTTYPE_SUCCESS == wtap_block_get_nth_string_option_value(block, OPT_COMMENT, i, &comment)) { lua_pushstring(L, comment); } else { lua_pushnil(L); } lua_settable(L, FRAMEINFO_COMMENTS_TABLE); } return 1; } static int FrameInfo_set_comment (lua_State* L) { FrameInfo fi = checkFrameInfo(L,1); #define FRAMEINFO_COMMENTS_NEWTABLE 2 #define FRAMEINFO_COMMENTS_NEWCOMMENT 2 size_t len = 0; char *comment = NULL; wtap_block_t block = NULL; unsigned i = 0; unsigned n_comments = 0; if(fi->rec->block != NULL) { block = fi->rec->block; } else { block = wtap_block_create(WTAP_BLOCK_PACKET); fi->rec->block = block; } /* Strip off old comments */ n_comments = wtap_block_count_option(block, OPT_COMMENT); for (i = 0; i < n_comments; i++) { wtap_block_remove_nth_option_instance(block, OPT_COMMENT, 0); } /* Add new comment(s) */ if (lua_istable(L, FRAMEINFO_COMMENTS_NEWTABLE)) { for (lua_pushnil(L); lua_next(L, FRAMEINFO_COMMENTS_NEWTABLE); ) { if (lua_isstring(L,-1)) { comment = (char *)luaL_checklstring(L,-1,&len); wtap_block_add_string_option(block, OPT_COMMENT, comment, len); } else if (! lua_isnil(L,-1) ) { return luaL_error(L,"only strings should be in the table"); } lua_pop(L, 1); } } else if (lua_isstring(L, FRAMEINFO_COMMENTS_NEWCOMMENT)) { comment = (char *)luaL_checklstring(L,FRAMEINFO_COMMENTS_NEWCOMMENT,&len); wtap_block_add_string_option(block, OPT_COMMENT, comment, len); } else { return luaL_error(L,"comment must be either a string or an array of strings"); } return 0; } /* WSLUA_ATTRIBUTE FrameInfo_time RW The packet timestamp as an NSTime object. Note: Set the `FileHandler.time_precision` to the appropriate `wtap_file_tsprec` value as well. */ static int FrameInfo_set_time (lua_State* L) { FrameInfo fi = checkFrameInfo(L,1); NSTime nstime = checkNSTime(L,2); if (!fi->rec) return 0; fi->rec->ts.secs = nstime->secs; fi->rec->ts.nsecs = nstime->nsecs; return 0; } static int FrameInfo_get_time (lua_State* L) { FrameInfo fi = checkFrameInfo(L,1); NSTime nstime = (NSTime)g_malloc(sizeof(nstime_t)); if (!nstime) return 0; nstime->secs = fi->rec->ts.secs; nstime->nsecs = fi->rec->ts.nsecs; pushNSTime(L,nstime); return 1; /* An NSTime object of the frame's timestamp. */ } /* WSLUA_ATTRIBUTE FrameInfo_data RW The data buffer containing the packet. [NOTE] ==== This cannot be cleared once set. ==== */ static int FrameInfo_set_data (lua_State* L) { FrameInfo fi = checkFrameInfo(L,1); if (!fi->rec) { ws_warning("Error in FrameInfo set data: NULL pointer"); return 0; } if (!fi->buf) { ws_warning("Error in FrameInfo set data: NULL frame_buffer pointer"); return 0; } if (lua_isstring(L,2)) { size_t len = 0; const char* s = luaL_checklstring(L,2,&len); /* Make sure we have enough room for the packet */ ws_buffer_assure_space(fi->buf, len); memcpy(ws_buffer_start_ptr(fi->buf), s, len); fi->rec->rec_header.packet_header.caplen = (uint32_t) len; fi->rec->rec_header.packet_header.len = (uint32_t) len; } else luaL_error(L, "FrameInfo's attribute 'data' must be a Lua string"); return 0; } static int FrameInfo_get_data (lua_State* L) { FrameInfo fi = checkFrameInfo(L,1); if (!fi->buf) return 0; lua_pushlstring(L, ws_buffer_start_ptr(fi->buf), ws_buffer_length(fi->buf)); WSLUA_RETURN(1); /* A Lua string of the frame buffer's data. */ } /* WSLUA_ATTRIBUTE FrameInfo_rec_type RW The record type of the packet frame See `wtap_rec_types` for values. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfo,rec_type,rec->rec_type); WSLUA_ATTRIBUTE_NAMED_INTEGER_SETTER(FrameInfo,rec_type,rec->rec_type,unsigned); /* WSLUA_ATTRIBUTE FrameInfo_flags RW The presence flags of the packet frame. See `wtap_presence_flags` for bit values. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfo,flags,rec->presence_flags); WSLUA_ATTRIBUTE_NAMED_INTEGER_SETTER(FrameInfo,flags,rec->presence_flags,uint32_t); /* WSLUA_ATTRIBUTE FrameInfo_captured_length RW The captured packet length, and thus the length of the buffer passed to the `FrameInfo.data` field. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfo,captured_length,rec->rec_header.packet_header.caplen); WSLUA_ATTRIBUTE_NAMED_INTEGER_SETTER(FrameInfo,captured_length,rec->rec_header.packet_header.caplen,uint32_t); /* WSLUA_ATTRIBUTE FrameInfo_original_length RW The on-the-wire packet length, which may be longer than the `captured_length`. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfo,original_length,rec->rec_header.packet_header.len); WSLUA_ATTRIBUTE_NAMED_INTEGER_SETTER(FrameInfo,original_length,rec->rec_header.packet_header.len,uint32_t); /* WSLUA_ATTRIBUTE FrameInfo_encap RW The packet encapsulation type for the frame/packet, if the file supports per-packet types. See `wtap_encaps` for possible packet encapsulation types to use as the value for this field. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfo,encap,rec->rec_header.packet_header.pkt_encap); WSLUA_ATTRIBUTE_NAMED_INTEGER_SETTER(FrameInfo,encap,rec->rec_header.packet_header.pkt_encap,int); /* This table is ultimately registered as a sub-table of the class' metatable, * and if __index/__newindex is invoked then it calls the appropriate function * from this table for getting/setting the members. */ WSLUA_ATTRIBUTES FrameInfo_attributes[] = { WSLUA_ATTRIBUTE_RWREG(FrameInfo,rec_type), WSLUA_ATTRIBUTE_RWREG(FrameInfo,flags), WSLUA_ATTRIBUTE_RWREG(FrameInfo,captured_length), WSLUA_ATTRIBUTE_RWREG(FrameInfo,original_length), WSLUA_ATTRIBUTE_RWREG(FrameInfo,comment), WSLUA_ATTRIBUTE_RWREG(FrameInfo,encap), WSLUA_ATTRIBUTE_RWREG(FrameInfo,time), WSLUA_ATTRIBUTE_RWREG(FrameInfo,data), { NULL, NULL, NULL } }; WSLUA_METHODS FrameInfo_methods[] = { WSLUA_CLASS_FNREG(FrameInfo,read_data), { NULL, NULL } }; WSLUA_META FrameInfo_meta[] = { WSLUA_CLASS_MTREG(FrameInfo,tostring), { NULL, NULL } }; int FrameInfo_register(lua_State* L) { WSLUA_REGISTER_CLASS_WITH_ATTRS(FrameInfo); return 0; } WSLUA_CLASS_DEFINE(FrameInfoConst,FAIL_ON_NULL_OR_EXPIRED("FrameInfo")); /* A constant FrameInfo object, passed into Lua as an argument by the FileHandler write callback function. This has similar attributes/properties as FrameInfo, but the fields can only be read from, not written to. */ FrameInfoConst* push_FrameInfoConst(lua_State* L, const wtap_rec *rec, const uint8_t *pd) { FrameInfoConst f = (FrameInfoConst) g_malloc(sizeof(struct _wslua_const_phdr)); f->rec = rec; f->pd = pd; f->expired = false; return pushFrameInfoConst(L,f); } WSLUA_METAMETHOD FrameInfoConst__tostring(lua_State* L) { /* Generates a string of debug info for the FrameInfo */ FrameInfoConst fi = toFrameInfoConst(L,1); if (!fi) { lua_pushstring(L,"FrameInfo pointer is NULL!"); } else { if (fi->rec && !fi->expired) lua_pushfstring(L, "FrameInfo: rec_type=%u, presence_flags=%d, caplen=%d, len=%d, pkt_encap=%d, block='%p'", fi->rec->rec_type, fi->rec->presence_flags, fi->rec->rec_header.packet_header.caplen, fi->rec->rec_header.packet_header.len, fi->rec->rec_header.packet_header.pkt_encap, fi->rec->block); else lua_pushfstring(L, "FrameInfo has %s", fi->rec?"expired":"null rec pointer"); } WSLUA_RETURN(1); /* String of debug information. */ } /* XXX: should this function be a method of File instead? */ WSLUA_METHOD FrameInfoConst_write_data(lua_State* L) { /* Tells Wireshark to write directly to given file from the frame data buffer, for length bytes. Returns true if succeeded, else false. */ #define WSLUA_ARG_FrameInfoConst_write_data_FILE 2 /* The File object userdata, provided by Wireshark previously in a writing-based callback. */ #define WSLUA_OPTARG_FrameInfoConst_write_data_LENGTH 3 /* The number of bytes to write to the file at the current cursor position, or all if not supplied. */ FrameInfoConst fi = checkFrameInfoConst(L,1); File fh = checkFile(L,WSLUA_ARG_FrameInfoConst_write_data_FILE); uint32_t len = wslua_optuint32(L, WSLUA_OPTARG_FrameInfoConst_write_data_LENGTH, fi->rec ? fi->rec->rec_header.packet_header.caplen:0); int err = 0; if (!fi->pd || !fi->rec || !fh->wdh) { luaL_error(L, "FrameInfoConst write_data() got null buffer or file pointer internally"); return 0; } if (len > fi->rec->rec_header.packet_header.caplen) len = fi->rec->rec_header.packet_header.caplen; if (!wtap_dump_file_write(fh->wdh, fi->pd, (size_t)(len), &err)) { lua_pushboolean(L, false); lua_pushfstring(L, "FrameInfoConst write_data() error: %s", g_strerror(err)); lua_pushinteger(L, err); return 3; } lua_pushboolean(L, true); WSLUA_RETURN(1); /* True if succeeded, else returns false along with the error number and string error description. */ } /* free the struct we created, but not the wtap_rec it points to */ static int FrameInfoConst__gc(lua_State* L) { FrameInfoConst fi = toFrameInfoConst(L,1); g_free(fi); return 0; } /* WSLUA_ATTRIBUTE FrameInfoConst_comment RO The first string comment for the packet, if any; nil if there is no comment. */ static int FrameInfoConst_get_comment (lua_State* L) { FrameInfoConst fi = checkFrameInfoConst(L,1); #define FRAMEINFOCONST_COMMENTS_TABLE 2 char *comment = NULL; wtap_block_t block = NULL; unsigned i = 0; unsigned n_comments = 0; block = fi->rec->block; // XXX - how to get the user-edited block, if any? n_comments = wtap_block_count_option(block, OPT_COMMENT); lua_createtable(L, n_comments, 0); for (i = 0; i < n_comments; i++) { comment = NULL; lua_pushinteger(L, i+1); if (WTAP_OPTTYPE_SUCCESS == wtap_block_get_nth_string_option_value(block, OPT_COMMENT, i, &comment)) { lua_pushstring(L, comment); } else { lua_pushnil(L); } lua_settable(L, FRAMEINFOCONST_COMMENTS_TABLE); } return 1; } /* WSLUA_ATTRIBUTE FrameInfoConst_time RO The packet timestamp as an NSTime object. */ static int FrameInfoConst_get_time (lua_State* L) { FrameInfoConst fi = checkFrameInfoConst(L,1); NSTime nstime = (NSTime)g_malloc(sizeof(nstime_t)); if (!nstime) return 0; nstime->secs = fi->rec->ts.secs; nstime->nsecs = fi->rec->ts.nsecs; pushNSTime(L,nstime); return 1; /* An NSTime object of the frame's timestamp. */ } /* WSLUA_ATTRIBUTE FrameInfoConst_data RO The data buffer containing the packet. */ static int FrameInfoConst_get_data (lua_State* L) { FrameInfoConst fi = checkFrameInfoConst(L,1); if (!fi->pd || !fi->rec) return 0; lua_pushlstring(L, fi->pd, fi->rec->rec_header.packet_header.caplen); return 1; } /* WSLUA_ATTRIBUTE FrameInfoConst_rec_type RO The record type of the packet frame - see `wtap_presence_flags` for values. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfoConst,rec_type,rec->rec_type); /* WSLUA_ATTRIBUTE FrameInfoConst_flags RO The presence flags of the packet frame - see `wtap_presence_flags` for bits. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfoConst,flags,rec->presence_flags); /* WSLUA_ATTRIBUTE FrameInfoConst_captured_length RO The captured packet length, and thus the length of the buffer in the FrameInfoConst.data field. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfoConst,captured_length,rec->rec_header.packet_header.caplen); /* WSLUA_ATTRIBUTE FrameInfoConst_original_length RO The on-the-wire packet length, which may be longer than the `captured_length`. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfoConst,original_length,rec->rec_header.packet_header.len); /* WSLUA_ATTRIBUTE FrameInfoConst_encap RO The packet encapsulation type, if the file supports per-packet types. See `wtap_encaps` for possible packet encapsulation types to use as the value for this field. */ WSLUA_ATTRIBUTE_NAMED_INTEGER_GETTER(FrameInfoConst,encap,rec->rec_header.packet_header.pkt_encap); WSLUA_ATTRIBUTES FrameInfoConst_attributes[] = { WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,rec_type), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,flags), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,captured_length), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,original_length), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,encap), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,comment), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,time), WSLUA_ATTRIBUTE_ROREG(FrameInfoConst,data), { NULL, NULL, NULL } }; WSLUA_METHODS FrameInfoConst_methods[] = { WSLUA_CLASS_FNREG(FrameInfoConst,write_data), { NULL, NULL } }; WSLUA_META FrameInfoConst_meta[] = { WSLUA_CLASS_MTREG(FrameInfoConst,tostring), { NULL, NULL } }; int FrameInfoConst_register(lua_State* L) { WSLUA_REGISTER_CLASS_WITH_ATTRS(FrameInfoConst); return 0; } /* * Editor modelines - https://www.wireshark.org/tools/modelines.html * * Local variables: * c-basic-offset: 4 * tab-width: 8 * indent-tabs-mode: nil * End: * * vi: set shiftwidth=4 tabstop=8 expandtab: * :indentSize=4:tabSize=8:noTabs=true: */