blob: 0ec0bcc04fa43a9d032c79f7ca1e9b90b4f5603f [file] [log] [blame]
-- Copyright (c) 2015-2019, Regents of the University of California.
--
-- This file is part of ndn-tools (Named Data Networking Essential Tools).
-- See AUTHORS.md for complete list of ndn-tools authors and contributors.
--
-- ndn-tools is free software: you can redistribute it and/or modify it under the terms
-- of the GNU General Public License as published by the Free Software Foundation,
-- either version 3 of the License, or (at your option) any later version.
--
-- ndn-tools is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
-- without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
-- PURPOSE. See the GNU General Public License for more details.
--
-- You should have received a copy of the GNU General Public License along with
-- ndn-tools, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
--
-- @author Qi Zhao <https://www.linkedin.com/pub/qi-zhao/73/835/9a3>
-- @author Seunghyun Yoo <http://relue2718.com/>
-- @author Seungbae Kim <https://sites.google.com/site/sbkimcv/>
-- @author Alexander Afanasyev <http://lasr.cs.ucla.edu/afanasyev/index.html>
-- @author Zipeng Wang
-- @author Qianshan Yu
-- inspect.lua (https://github.com/kikito/inspect.lua) can be used for debugging.
-- See more at http://stackoverflow.com/q/15175859/2150331
-- local inspect = require('inspect')
-- NDN protocol
ndn = Proto("ndn", "Named Data Networking (NDN)")
-- TODO with NDNLPv2 processing:
-- * mark field "unknown" when the field is recognized but the relevant feature is disabled
-- * colorize "unknown field"
-- * for a field that appears out-of-order, display "out-of-order field " in red
-----------------------------------------------------
-----------------------------------------------------
-- Field formatting helpers
-- @return TLV-VALUE portion of a TLV block
function getValue(b)
return b.tvb(b.offset + b.typeLen + b.lengthLen, b.length)
end
function getNonNegativeInteger(b)
if b.length == 1 or b.length == 2 or b.length == 4 or b.length == 8 then
return getValue(b):uint64()
end
return UInt64.max()
end
function getUriFromSha256DigestComponent(b)
local s = ""
if b.type == 1 then
s = "sha256digest="
elseif b.type == 2 then
s = "params-sha256="
else
assert(false)
end
for i = 0, (b.length - 1) do
local byte = b.tvb(b.offset + b.typeLen + b.lengthLen + i, 1)
s = s .. string.format("%02x", byte:uint())
end
return s
end
function getUriFromNameComponent(b)
if b.type == 1 or b.type == 2 then
return getUriFromSha256DigestComponent(b)
end
local s = ""
if b.type ~= 8 then
s = string.format("%d=", b.type)
end
hasNonPeriod = false
for i = 0, (b.length - 1) do
local byte = b.tvb(b.offset + b.typeLen + b.lengthLen + i, 1)
local ch = byte:uint()
hasNonPeriod = hasNonPeriod or ch ~= 0x2E
if (ch >= 0x41 and ch <= 0x5A) or (ch >= 0x61 and ch <= 0x7A) or (ch >= 0x30 and ch <= 0x39) or ch == 0x2D or ch == 0x2E or ch == 0x5F or ch == 0x7E then
s = s .. byte:string()
else
s = s .. string.format("%%%02X", ch)
end
end
if not hasNonPeriod then
s = s .. "..."
end
return s
end
function getUriFromName(b)
if b.elements == nil then
return "/"
end
components = {}
for i, comp in pairs(b.elements) do
table.insert(components, getUriFromNameComponent(comp))
end
return "/" .. table.concat(components, "/")
end
function getUriFromFinalBlockId(b)
if b.elements == nil then
return "/"
end
return getUriFromNameComponent(b.elements[1])
end
function getNackReasonDetail(b)
assert(b.type == 801)
local code = getNonNegativeInteger(b)
if code == UInt64(0) then return "None"
elseif code == UInt64(50) then return "Congestion"
elseif code == UInt64(100) then return "Duplicate"
elseif code == UInt64(150) then return "NoRoute"
else return tostring(code)
end
end
function getCachePolicyDetail(b)
assert(b.type == 821)
local code = getNonNegativeInteger(b)
if code == UInt64(0) then return "None"
elseif code == UInt64(1) then return "NoCache"
else return tostring(code) .. " (unknown)"
end
end
function getNonce(b)
assert(b.type == 10)
if b.length ~= 4 then
return "invalid (should have 4 octets)"
end
return getValue(b):uint()
end
function getHopLimit(b)
assert(b.type == 34)
if b.length ~= 1 then
return "invalid (should have 1 octet)"
end
return getValue(b):uint()
end
function getTrue(block)
return "Yes"
end
local AppPrivateBlock1 = 100
local AppPrivateBlock2 = 800
local AppPrivateBlock3 = 1000
function canIgnoreTlvType(t)
if t < AppPrivateBlock2 or t >= AppPrivateBlock3 then
return false
else
if math.fmod(t, 2) == 1 then
return true
else
return false
end
end
end
function getGenericBlockInfo(block)
local name = ""
-- TODO: Properly format informational message based type value reservations
-- (https://named-data.net/doc/NDN-packet-spec/current/types.html#tlv-type-number-reservations)
if (block.type <= AppPrivateBlock1) then
name = "Unrecognized from the reserved range " .. 0 .. "-" .. AppPrivateBlock1 .. ""
elseif (AppPrivateBlock1 < block.type and block.type < AppPrivateBlock2) then
name = "Unrecognized from the reserved range " .. (AppPrivateBlock1 + 1) .. "-" .. (AppPrivateBlock2 - 1) .. ""
elseif (AppPrivateBlock2 <= block.type and block.type <= AppPrivateBlock3) then
if canIgnoreTlvType(block.type) then
name = "Unknown field (ignored)"
else
name = "Unknown field"
end
else
name = "RESERVED_3"
end
return name .. ", Type: " .. block.type .. ", Length: " .. block.length
end
-----------------------------------------------------
-----------------------------------------------------
local NDN_DICT = {
-- Name and name components
[7] = {name = "Name" , field = ProtoField.string("ndn.name", "Name") , value = getUriFromName},
[1] = {name = "ImplicitSha256DigestComponent" , field = ProtoField.string("ndn.implicit_sha256", "ImplicitSha256DigestComponent"), value = getUriFromNameComponent},
[2] = {name = "ParametersSha256DigestComponent", field = ProtoField.string("ndn.params_sha256", "ParametersSha256DigestComponent"), value = getUriFromNameComponent},
[8] = {name = "GenericNameComponent" , field = ProtoField.string("ndn.namecomponent", "GenericNameComponent") , value = getUriFromNameComponent},
-- Interest and its sub-elements in Packet Format v0.3
[5] = {name = "Interest" , summary = true},
[33] = {name = "CanBePrefix" , field = ProtoField.string("ndn.canbeprefix", "CanBePrefix") , value = getTrue},
[18] = {name = "MustBeFresh" , field = ProtoField.string("ndn.mustbefresh", "MustBeFresh") , value = getTrue},
-- [30] = {name = "ForwardingHint" , summary = true},
-- ForwardingHint and LinkPreference have the same TLV-TYPE number, so we can't recognize both for now (see #4185).
[31] = {name = "LinkDelegation" , summary = true},
[30] = {name = "LinkPreference" , field = ProtoField.uint64("ndn.linkpreference", "LinkPreference", base.DEC) , value = getNonNegativeInteger},
[10] = {name = "Nonce" , field = ProtoField.uint32("ndn.nonce", "Nonce", base.HEX) , value = getNonce},
[12] = {name = "InterestLifetime" , field = ProtoField.uint64("ndn.lifetime", "InterestLifetime", base.DEC) , value = getNonNegativeInteger},
[34] = {name = "HopLimit" , field = ProtoField.uint8("ndn.hoplimit", "HopLimit", base.DEC) , value = getHopLimit},
[36] = {name = "ApplicationParameters" , field = ProtoField.bytes("ndn.app_params", "ApplicationParameters")},
-- Data and its sub-elements in Packet Format v0.3
[6] = {name = "Data" , summary = true},
[20] = {name = "MetaInfo" , summary = true},
[24] = {name = "ContentType" , field = ProtoField.uint64("ndn.contenttype", "ContentType", base.DEC) , value = getNonNegativeInteger},
[25] = {name = "FreshnessPeriod" , field = ProtoField.uint64("ndn.freshness", "FreshnessPeriod", base.DEC) , value = getNonNegativeInteger},
[26] = {name = "FinalBlockId" , field = ProtoField.string("ndn.finalblock", "FinalBlockId") , value = getUriFromFinalBlockId},
[21] = {name = "Content" , field = ProtoField.bytes("ndn.content", "Content")},
[22] = {name = "SignatureInfo" , summary = true},
[27] = {name = "SignatureType" , field = ProtoField.uint64("ndn.sigtype", "SignatureType", base.DEC) , value = getNonNegativeInteger},
[28] = {name = "KeyLocator" , summary = true},
[29] = {name = "KeyDigest" , field = ProtoField.bytes("ndn.keydigest", "KeyDigest")},
[23] = {name = "SignatureValue" , field = ProtoField.bytes("ndn.sigvalue", "SignatureValue")},
-- NDNLPv2 headers
[80] = {name = "Fragment" },
[81] = {name = "Sequence" , field = ProtoField.uint64("ndn.sequence", "Sequence", base.DEC) , value = getNonNegativeInteger},
[82] = {name = "FragIndex" , field = ProtoField.uint64("ndn.fragindex", "FragIndex", base.DEC) , value = getNonNegativeInteger},
[83] = {name = "FragCount" , field = ProtoField.uint64("ndn.fragcount", "FragCount", base.DEC) , value = getNonNegativeInteger},
[98] = {name = "PitToken" , field = ProtoField.bytes("ndn.pit_token", "PitToken")},
[100] = {name = "LpPacket" , summary = true},
[800] = {name = "Nack" , summary = true},
[801] = {name = "NackReason" , field = ProtoField.string("ndn.nack_reason", "NackReason") , value = getNackReasonDetail},
[812] = {name = "IncomingFaceId" , field = ProtoField.uint64("ndn.incoming_face", "IncomingFaceId", base.DEC) , value = getNonNegativeInteger},
[816] = {name = "NextHopFaceId" , field = ProtoField.uint64("ndn.nexthop_face", "NextHopFaceId", base.DEC) , value = getNonNegativeInteger},
[820] = {name = "CachePolicy" , summary = true},
[821] = {name = "CachePolicyType" , field = ProtoField.string("ndn.cache_policy", "CachePolicyType") , value = getCachePolicyDetail},
[832] = {name = "CongestionMark" , field = ProtoField.uint64("ndn.congestion_mark", "CongestionMark", base.DEC) , value = getNonNegativeInteger},
[836] = {name = "Ack" , field = ProtoField.uint64("ndn.ack", "Ack", base.DEC) , value = getNonNegativeInteger},
[840] = {name = "TxSequence" , field = ProtoField.uint64("ndn.txseq", "TxSequence", base.DEC) , value = getNonNegativeInteger},
}
-- -- Add protofields in NDN protocol
ndn.fields = {}
for key, value in pairs(NDN_DICT) do
table.insert(ndn.fields, value.field)
end
-----------------------------------------------------
-----------------------------------------------------
-- block
-- .tvb
-- .offset
-- .type
-- .typeLen
-- .length
-- .lengthLen
-- .size = .typeLen + .lengthLen + .length
function addInfo(block, root) -- may be add additional context later
local info = NDN_DICT[block.type]
if (info == nil) then
info = {}
info.value = getGenericBlockInfo
-- color
end
local treeInfo
if (info.value == nil) then
if (info.field ~= nil) then
treeInfo = root:add(info.field, block.tvb(block.offset, block.size))
else
treeInfo = root:add(block.tvb(block.offset, block.size), info.name)
end
treeInfo:append_text(", Type: " .. block.type .. ", Length: " .. block.length)
else
block.value = info.value(block)
if (info.field ~= nil) then
treeInfo = root:add(info.field, block.tvb(block.offset, block.size), block.value)
else
treeInfo = root:add(block.tvb(block.offset, block.size), block.value)
end
end
block.root = treeInfo
return block.root
end
function addSummary(block)
if (block.elements == nil) then
return
end
local info = NDN_DICT[block.type]
if (info == nil or info.summary == nil) then
return
end
local summary = {}
for k, subblock in pairs(block.elements) do
if (subblock.value ~= nil) then
local info = NDN_DICT[subblock.type]
if (info ~= nil) then
table.insert(summary, info.name .. ": " .. subblock.value)
end
end
end
if (#summary > 0) then
block.summary = table.concat(summary, ", ")
if (block.value == nil) then
block.value = block.summary
end
block.root:append_text(", " .. block.summary)
end
end
-----------------------------------------------------
-----------------------------------------------------
function readVarNumber(tvb, offset)
if offset >= tvb:len() then
return 0, 0
end
local firstOctet = tvb(offset, 1):uint()
if (firstOctet < 253) then
return firstOctet, 1
elseif (firstOctet == 253) and (offset + 3 < tvb:len()) then
return tvb(offset + 1, 2):uint(), 3
elseif (firstOctet == 254) and (offset + 5 < tvb:len()) then
return tvb(offset + 1, 4):uint(), 5
elseif (firstOctet == 255) and (offset + 9 < tvb:len()) then
return tvb(offset + 1, 8):uint64(), 9
end
return 0, 0
end
function getBlock(tvb, offset)
local block = {}
block.tvb = tvb
block.offset = offset
block.type, block.typeLen = readVarNumber(block.tvb, block.offset)
if block.typeLen == 0 then
return nil
end
block.length, block.lengthLen = readVarNumber(block.tvb, block.offset + block.typeLen)
if block.lengthLen == 0 then
return nil
end
block.size = block.typeLen + block.lengthLen + block.length
return block
end
function canBeValidNdnPacket(block)
if ((block.type == 5 or block.type == 6 or block.type == 100) and block.length <= 8800) then
return true
else
return false
end
end
function findNdnPacket(tvb)
offset = 0
while offset + 2 < tvb:len() do
local block = getBlock(tvb, offset)
if (block ~= nil) and canBeValidNdnPacket(block) then
return block
end
offset = offset + 1
end
return nil
end
function getSubBlocks(block)
local valueLeft = block.length
local subBlocks = {}
while valueLeft > 0 do
local offset = block.offset + block.typeLen + block.lengthLen + (block.length - valueLeft)
local child = getBlock(block.tvb, offset)
if child == nil then
return nil
end
valueLeft = valueLeft - child.size
table.insert(subBlocks, child)
end
if (valueLeft == 0) then
return subBlocks
else
return nil
end
end
-----------------------------------------------------
-----------------------------------------------------
-- NDN protocol dissector function
function ndn.dissector(tvb, pInfo, root) -- Tvb, Pinfo, TreeItem
if (tvb:len() ~= tvb:reported_len()) then
return 0 -- ignore partially captured packets
-- this can/may be re-enabled only for unfragmented UDP packets
end
local ok, block = pcall(findNdnPacket, tvb)
if (not ok) then
return 0
end
if (block == nil or block.offset == nil) then
-- no valid NDN packets found
return 0
end
local nBytesLeft = tvb:len() - block.offset
-- print (pInfo.number .. ":: Found block: " .. block.type .. " of length " .. block.size .. " bytesLeft: " .. nBytesLeft)
local pktType = ""
local pktName = ""
while (block.size <= nBytesLeft) do
-- Create TreeItems
block.tree = root:add(ndn, tvb(block.offset, block.size))
local queue = {block}
while (#queue > 0) do
local block = queue[1]
table.remove(queue, 1)
block.elements = getSubBlocks(block)
local subtree = addInfo(block, block.tree)
if (block.elements ~= nil) then
for i, subBlock in pairs(block.elements) do
subBlock.tree = subtree
table.insert(queue, subBlock)
end
end
end
-- Make summaries
local queue = {block}
while (#queue > 0) do
local block = queue[1]
if (block.visited ~= nil or block.elements == nil) then
-- try to make summary
table.remove(queue, 1)
addSummary(block)
else
for i, subBlock in pairs(block.elements) do
table.insert(queue, 1, subBlock)
end
block.visited = true
end
-- prepare information to fill info column
if block.type == 5 and pktType ~= "Nack" then
pktType = "Interest"
elseif block.type == 6 then
pktType = "Data"
elseif block.type == 800 then
pktType = "Nack"
end
if block.type == 7 then
pktName = getUriFromName(block)
end
end
local info = NDN_DICT[block.type]
if (info ~= nil) then
if (block.summary ~= nil) then
block.tree:append_text(", " .. NDN_DICT[block.type].name .. ", " .. block.summary)
else
block.tree:append_text(", " .. NDN_DICT[block.type].name)
end
end
nBytesLeft = nBytesLeft - block.size
if (nBytesLeft > 0) then
ok, block = pcall(getBlock, tvb, tvb:len() - nBytesLeft)
if (not ok or block == nil or not canBeValidNdnPacket(block)) then
break
end
end
end -- while(block.size <= nBytesLeft)
pInfo.cols.protocol = tostring(pInfo.cols.protocol) .. " (" .. ndn.name .. ")"
pInfo.cols.info = pktType .. " " .. pktName
if (nBytesLeft > 0 and block ~= nil and block.size ~= nil and block.size > nBytesLeft) then
pInfo.desegment_offset = tvb:len() - nBytesLeft
-- Originally, I set desegment_len to the exact lenght, but it mysteriously didn't work for TCP
-- pInfo.desegment_len = block.size -- this will not work to desegment TCP streams
pInfo.desegment_len = DESEGMENT_ONE_MORE_SEGMENT
end
end
local udpDissectorTable = DissectorTable.get("udp.port")
udpDissectorTable:add("6363", ndn)
udpDissectorTable:add("56363", ndn)
local tcpDissectorTable = DissectorTable.get("tcp.port")
tcpDissectorTable:add("6363", ndn)
local websocketDissectorTable = DissectorTable.get("ws.port")
-- websocketDissectorTable:add("9696", ndn)
websocketDissectorTable:add("1-65535", ndn)
local ethernetDissectorTable = DissectorTable.get("ethertype")
ethernetDissectorTable:add(0x8624, ndn)
local pppDissectorTable = DissectorTable.get("ppp.protocol")
pppDissectorTable:add(0x0077, ndn)
io.stderr:write("NDN dissector successfully loaded\n")