You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
suricata/rust/src/smb/smb.rs

2178 lines
80 KiB
Rust

/* Copyright (C) 2017-2020 Open Information Security Foundation
*
* You can copy, redistribute or modify this Program under the terms of
* the GNU General Public License version 2 as published by the Free
* Software Foundation.
*
* This program 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
* version 2 along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
/* TODO
* - check all parsers for calls on non-SUCCESS status
*/
/* GAP processing:
* - if post-gap we've seen a succesful tx req/res: we consider "re-sync'd"
*/
// written by Victor Julien
use std;
use std::mem::transmute;
use std::str;
use std::ffi::{self, CStr};
use std::collections::HashMap;
use nom;
use crate::core::*;
use crate::applayer;
use crate::applayer::*;
use crate::filecontainer::*;
use crate::smb::nbss_records::*;
use crate::smb::smb1_records::*;
use crate::smb::smb2_records::*;
use crate::smb::smb1::*;
use crate::smb::smb2::*;
use crate::smb::smb3::*;
use crate::smb::dcerpc::*;
use crate::smb::session::*;
use crate::smb::events::*;
use crate::smb::files::*;
use crate::smb::smb2_ioctl::*;
pub const MIN_REC_SIZE: u16 = 32 + 4; // SMB hdr + nbss hdr
pub const SMB_CONFIG_DEFAULT_STREAM_DEPTH: u32 = 0;
static mut ALPROTO_SMB: AppProto = ALPROTO_UNKNOWN;
pub static mut SURICATA_SMB_FILE_CONFIG: Option<&'static SuricataFileContext> = None;
#[no_mangle]
pub extern "C" fn rs_smb_init(context: &'static mut SuricataFileContext)
{
unsafe {
SURICATA_SMB_FILE_CONFIG = Some(context);
}
}
pub const SMB_NTSTATUS_SUCCESS: u32 = 0;
pub const SMB_NTSTATUS_PENDING: u32 = 0x00000103;
pub const SMB_NTSTATUS_BUFFER_OVERFLOW: u32 = 0x80000005;
pub const SMB_NTSTATUS_NO_MORE_FILES: u32 = 0x80000006;
pub const SMB_NTSTATUS_NO_MORE_ENTRIES: u32 = 0x8000001a;
pub const SMB_NTSTATUS_INVALID_HANDLE: u32 = 0xc0000008;
pub const SMB_NTSTATUS_INVALID_PARAMETER: u32 = 0xc000000d;
pub const SMB_NTSTATUS_NO_SUCH_DEVICE: u32 = 0xc000000e;
pub const SMB_NTSTATUS_NO_SUCH_FILE: u32 = 0xc000000f;
pub const SMB_NTSTATUS_INVALID_DEVICE_REQUEST: u32 = 0xc0000010;
pub const SMB_NTSTATUS_END_OF_FILE: u32 = 0xc0000011;
pub const SMB_NTSTATUS_MORE_PROCESSING_REQUIRED: u32 = 0xc0000016;
pub const SMB_NTSTATUS_ACCESS_DENIED: u32 = 0xc0000022;
pub const SMB_NTSTATUS_OBJECT_NAME_INVALID: u32 = 0xc0000033;
pub const SMB_NTSTATUS_OBJECT_NAME_NOT_FOUND: u32 = 0xc0000034;
pub const SMB_NTSTATUS_OBJECT_NAME_COLLISION: u32 = 0xc0000035;
pub const SMB_NTSTATUS_OBJECT_PATH_NOT_FOUND: u32 = 0xc000003a;
pub const SMB_NTSTATUS_SHARING_VIOLATION: u32 = 0xc0000043;
pub const SMB_NTSTATUS_LOCK_CONFLICT: u32 = 0xc0000054;
pub const SMB_NTSTATUS_LOCK_NOT_GRANTED: u32 = 0xc0000055;
pub const SMB_NTSTATUS_PRIVILEGE_NOT_HELD: u32 = 0xc0000061;
pub const SMB_NTSTATUS_LOGON_FAILURE: u32 = 0xc000006d;
pub const SMB_NTSTATUS_PIPE_DISCONNECTED: u32 = 0xc00000b0;
pub const SMB_NTSTATUS_FILE_IS_A_DIRECTORY: u32 = 0xc00000ba;
pub const SMB_NTSTATUS_NOT_SUPPORTED: u32 = 0xc00000bb;
pub const SMB_NTSTATUS_BAD_NETWORK_NAME: u32 = 0xc00000cc;
pub const SMB_NTSTATUS_REQUEST_NOT_ACCEPTED: u32 = 0xc00000d0;
pub const SMB_NTSTATUS_OPLOCK_NOT_GRANTED: u32 = 0xc00000e2;
pub const SMB_NTSTATUS_CANCELLED: u32 = 0xc0000120;
pub const SMB_NTSTATUS_FILE_CLOSED: u32 = 0xc0000128;
pub const SMB_NTSTATUS_FS_DRIVER_REQUIRED: u32 = 0xc000019c;
pub const SMB_NTSTATUS_INSUFF_SERVER_RESOURCES: u32 = 0xc0000205;
pub const SMB_NTSTATUS_NOT_FOUND: u32 = 0xc0000225;
pub const SMB_NTSTATUS_PIPE_BROKEN: u32 = 0xc000014b;
pub const SMB_NTSTATUS_TRUSTED_RELATIONSHIP_FAILURE: u32 = 0xc000018d;
pub const SMB_NTSTATUS_NOT_A_REPARSE_POINT: u32 = 0xc0000275;
pub const SMB_NTSTATUS_NETWORK_SESSION_EXPIRED: u32 = 0xc000035c;
pub fn smb_ntstatus_string(c: u32) -> String {
match c {
SMB_NTSTATUS_SUCCESS => "STATUS_SUCCESS",
SMB_NTSTATUS_BUFFER_OVERFLOW => "STATUS_BUFFER_OVERFLOW",
SMB_NTSTATUS_PENDING => "STATUS_PENDING",
SMB_NTSTATUS_NO_MORE_FILES => "STATUS_NO_MORE_FILES",
SMB_NTSTATUS_NO_MORE_ENTRIES => "STATUS_NO_MORE_ENTRIES",
SMB_NTSTATUS_INVALID_HANDLE => "STATUS_INVALID_HANDLE",
SMB_NTSTATUS_INVALID_PARAMETER => "STATUS_INVALID_PARAMETER",
SMB_NTSTATUS_NO_SUCH_DEVICE => "STATUS_NO_SUCH_DEVICE",
SMB_NTSTATUS_NO_SUCH_FILE => "STATUS_NO_SUCH_FILE",
SMB_NTSTATUS_INVALID_DEVICE_REQUEST => "STATUS_INVALID_DEVICE_REQUEST",
SMB_NTSTATUS_END_OF_FILE => "STATUS_END_OF_FILE",
SMB_NTSTATUS_MORE_PROCESSING_REQUIRED => "STATUS_MORE_PROCESSING_REQUIRED",
SMB_NTSTATUS_ACCESS_DENIED => "STATUS_ACCESS_DENIED",
SMB_NTSTATUS_OBJECT_NAME_INVALID => "STATUS_OBJECT_NAME_INVALID",
SMB_NTSTATUS_OBJECT_NAME_NOT_FOUND => "STATUS_OBJECT_NAME_NOT_FOUND",
SMB_NTSTATUS_OBJECT_NAME_COLLISION => "STATUS_OBJECT_NAME_COLLISION",
SMB_NTSTATUS_OBJECT_PATH_NOT_FOUND => "STATUS_OBJECT_PATH_NOT_FOUND",
SMB_NTSTATUS_SHARING_VIOLATION => "STATUS_SHARING_VIOLATION",
SMB_NTSTATUS_LOCK_CONFLICT => "STATUS_LOCK_CONFLICT",
SMB_NTSTATUS_LOCK_NOT_GRANTED => "STATUS_LOCK_NOT_GRANTED",
SMB_NTSTATUS_PRIVILEGE_NOT_HELD => "STATUS_PRIVILEGE_NOT_HELD",
SMB_NTSTATUS_LOGON_FAILURE => "STATUS_LOGON_FAILURE",
SMB_NTSTATUS_PIPE_DISCONNECTED => "STATUS_PIPE_DISCONNECTED",
SMB_NTSTATUS_FILE_IS_A_DIRECTORY => "STATUS_FILE_IS_A_DIRECTORY",
SMB_NTSTATUS_NOT_SUPPORTED => "STATUS_NOT_SUPPORTED",
SMB_NTSTATUS_BAD_NETWORK_NAME => "STATUS_BAD_NETWORK_NAME",
SMB_NTSTATUS_REQUEST_NOT_ACCEPTED => "STATUS_REQUEST_NOT_ACCEPTED",
SMB_NTSTATUS_OPLOCK_NOT_GRANTED => "STATUS_OPLOCK_NOT_GRANTED",
SMB_NTSTATUS_CANCELLED => "STATUS_CANCELLED",
SMB_NTSTATUS_FILE_CLOSED => "STATUS_FILE_CLOSED",
SMB_NTSTATUS_FS_DRIVER_REQUIRED => "STATUS_FS_DRIVER_REQUIRED",
SMB_NTSTATUS_INSUFF_SERVER_RESOURCES => "STATUS_INSUFF_SERVER_RESOURCES",
SMB_NTSTATUS_NOT_FOUND => "STATUS_NOT_FOUND",
SMB_NTSTATUS_PIPE_BROKEN => "STATUS_PIPE_BROKEN",
SMB_NTSTATUS_TRUSTED_RELATIONSHIP_FAILURE => "STATUS_TRUSTED_RELATIONSHIP_FAILURE",
SMB_NTSTATUS_NOT_A_REPARSE_POINT => "STATUS_NOT_A_REPARSE_POINT",
SMB_NTSTATUS_NETWORK_SESSION_EXPIRED => "STATUS_NETWORK_SESSION_EXPIRED",
_ => { return (c).to_string(); },
}.to_string()
}
pub const SMB_SRV_ERROR: u16 = 1;
pub const SMB_SRV_BADPW: u16 = 2;
pub const SMB_SRV_BADTYPE: u16 = 3;
pub const SMB_SRV_ACCESS: u16 = 4;
pub const SMB_SRV_BADUID: u16 = 91;
pub fn smb_srv_error_string(c: u16) -> String {
match c {
SMB_SRV_ERROR => "SRV_ERROR",
SMB_SRV_BADPW => "SRV_BADPW",
SMB_SRV_BADTYPE => "SRV_BADTYPE",
SMB_SRV_ACCESS => "SRV_ACCESS",
SMB_SRV_BADUID => "SRV_BADUID",
_ => { return (c).to_string(); },
}.to_string()
}
pub const SMB_DOS_SUCCESS: u16 = 0;
pub const SMB_DOS_BAD_FUNC: u16 = 1;
pub const SMB_DOS_BAD_FILE: u16 = 2;
pub const SMB_DOS_BAD_PATH: u16 = 3;
pub const SMB_DOS_TOO_MANY_OPEN_FILES: u16 = 4;
pub const SMB_DOS_ACCESS_DENIED: u16 = 5;
pub fn smb_dos_error_string(c: u16) -> String {
match c {
SMB_DOS_SUCCESS => "DOS_SUCCESS",
SMB_DOS_BAD_FUNC => "DOS_BAD_FUNC",
SMB_DOS_BAD_FILE => "DOS_BAD_FILE",
SMB_DOS_BAD_PATH => "DOS_BAD_PATH",
SMB_DOS_TOO_MANY_OPEN_FILES => "DOS_TOO_MANY_OPEN_FILES",
SMB_DOS_ACCESS_DENIED => "DOS_ACCESS_DENIED",
_ => { return (c).to_string(); },
}.to_string()
}
pub const NTLMSSP_NEGOTIATE: u32 = 1;
pub const NTLMSSP_CHALLENGE: u32 = 2;
pub const NTLMSSP_AUTH: u32 = 3;
pub fn ntlmssp_type_string(c: u32) -> String {
match c {
NTLMSSP_NEGOTIATE => "NTLMSSP_NEGOTIATE",
NTLMSSP_CHALLENGE => "NTLMSSP_CHALLENGE",
NTLMSSP_AUTH => "NTLMSSP_AUTH",
_ => { return (c).to_string(); },
}.to_string()
}
#[derive(Default, Eq, PartialEq, Debug, Clone)]
pub struct SMBVerCmdStat {
smb_ver: u8,
smb1_cmd: u8,
smb2_cmd: u16,
status_set: bool,
status_is_dos_error: bool,
status_error_class: u8,
status: u32,
}
impl SMBVerCmdStat {
pub fn new() -> Self {
Default::default()
}
pub fn new1(cmd: u8) -> Self {
return Self {
smb_ver: 1,
smb1_cmd: cmd,
..Default::default()
}
}
pub fn new1_with_ntstatus(cmd: u8, status: u32) -> Self {
return Self {
smb_ver: 1,
smb1_cmd: cmd,
status_set: true,
status: status,
..Default::default()
}
}
pub fn new2(cmd: u16) -> Self {
return Self {
smb_ver: 2,
smb2_cmd: cmd,
..Default::default()
}
}
pub fn new2_with_ntstatus(cmd: u16, status: u32) -> Self {
return Self {
smb_ver: 2,
smb2_cmd: cmd,
status_set: true,
status: status,
..Default::default()
}
}
pub fn set_smb1_cmd(&mut self, cmd: u8) -> bool {
if self.smb_ver != 0 {
return false;
}
self.smb_ver = 1;
self.smb1_cmd = cmd;
return true;
}
pub fn set_smb2_cmd(&mut self, cmd: u16) -> bool {
if self.smb_ver != 0 {
return false;
}
self.smb_ver = 2;
self.smb2_cmd = cmd;
return true;
}
pub fn get_version(&self) -> u8 {
self.smb_ver
}
pub fn get_smb1_cmd(&self) -> (bool, u8) {
if self.smb_ver != 1 {
return (false, 0);
}
return (true, self.smb1_cmd);
}
pub fn get_smb2_cmd(&self) -> (bool, u16) {
if self.smb_ver != 2 {
return (false, 0);
}
return (true, self.smb2_cmd);
}
pub fn get_ntstatus(&self) -> (bool, u32) {
(self.status_set && !self.status_is_dos_error, self.status)
}
pub fn get_dos_error(&self) -> (bool, u8, u16) {
(self.status_set && self.status_is_dos_error, self.status_error_class, self.status as u16)
}
fn set_status(&mut self, status: u32, is_dos_error: bool)
{
if is_dos_error {
self.status_is_dos_error = true;
self.status_error_class = (status & 0x0000_00ff) as u8;
self.status = (status & 0xffff_0000) >> 16;
} else {
self.status = status;
}
self.status_set = true;
}
pub fn set_ntstatus(&mut self, status: u32)
{
self.set_status(status, false)
}
pub fn set_status_dos_error(&mut self, status: u32)
{
self.set_status(status, true)
}
}
/// "The FILETIME structure is a 64-bit value that represents the number of
/// 100-nanosecond intervals that have elapsed since January 1, 1601,
/// Coordinated Universal Time (UTC)."
#[derive(Eq, PartialEq, Debug, Clone)]
pub struct SMBFiletime {
ts: u64,
}
impl SMBFiletime {
pub fn new(raw: u64) -> Self {
Self {
ts: raw,
}
}
/// inspired by Bro, convert FILETIME to secs since unix epoch
pub fn as_unix(&self) -> u32 {
if self.ts > 116_444_736_000_000_000_u64 {
let ts = self.ts / 10000000 - 11644473600;
ts as u32
} else {
0
}
}
}
#[derive(Debug)]
pub enum SMBTransactionTypeData {
FILE(SMBTransactionFile),
TREECONNECT(SMBTransactionTreeConnect),
NEGOTIATE(SMBTransactionNegotiate),
DCERPC(SMBTransactionDCERPC),
CREATE(SMBTransactionCreate),
SESSIONSETUP(SMBTransactionSessionSetup),
IOCTL(SMBTransactionIoctl),
RENAME(SMBTransactionRename),
SETFILEPATHINFO(SMBTransactionSetFilePathInfo),
}
// Used for Trans2 SET_PATH_INFO and SET_FILE_INFO
#[derive(Debug)]
pub struct SMBTransactionSetFilePathInfo {
pub subcmd: u16,
pub loi: u16,
pub delete_on_close: bool,
pub filename: Vec<u8>,
pub fid: Vec<u8>,
}
impl SMBTransactionSetFilePathInfo {
pub fn new(filename: Vec<u8>, fid: Vec<u8>, subcmd: u16, loi: u16, delete_on_close: bool)
-> Self
{
return Self {
filename: filename, fid: fid,
subcmd: subcmd,
loi: loi,
delete_on_close: delete_on_close,
}
}
}
impl SMBState {
pub fn new_setfileinfo_tx(&mut self, filename: Vec<u8>, fid: Vec<u8>,
subcmd: u16, loi: u16, delete_on_close: bool)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
tx.type_data = Some(SMBTransactionTypeData::SETFILEPATHINFO(
SMBTransactionSetFilePathInfo::new(
filename, fid, subcmd, loi, delete_on_close)));
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
SCLogDebug!("SMB: TX SETFILEPATHINFO created: ID {}", tx.id);
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
pub fn new_setpathinfo_tx(&mut self, filename: Vec<u8>,
subcmd: u16, loi: u16, delete_on_close: bool)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
let fid : Vec<u8> = Vec::new();
tx.type_data = Some(SMBTransactionTypeData::SETFILEPATHINFO(
SMBTransactionSetFilePathInfo::new(filename, fid,
subcmd, loi, delete_on_close)));
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
SCLogDebug!("SMB: TX SETFILEPATHINFO created: ID {}", tx.id);
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
}
#[derive(Debug)]
pub struct SMBTransactionRename {
pub oldname: Vec<u8>,
pub newname: Vec<u8>,
pub fuid: Vec<u8>,
}
impl SMBTransactionRename {
pub fn new(fuid: Vec<u8>, oldname: Vec<u8>, newname: Vec<u8>) -> Self {
return Self {
fuid: fuid, oldname: oldname, newname: newname,
}
}
}
impl SMBState {
pub fn new_rename_tx(&mut self, fuid: Vec<u8>, oldname: Vec<u8>, newname: Vec<u8>)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
tx.type_data = Some(SMBTransactionTypeData::RENAME(
SMBTransactionRename::new(fuid, oldname, newname)));
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
SCLogDebug!("SMB: TX RENAME created: ID {}", tx.id);
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
}
#[derive(Default, Debug)]
pub struct SMBTransactionCreate {
pub disposition: u32,
pub delete_on_close: bool,
pub directory: bool,
pub filename: Vec<u8>,
pub guid: Vec<u8>,
pub create_ts: u32,
pub last_access_ts: u32,
pub last_write_ts: u32,
pub last_change_ts: u32,
pub size: u64,
}
impl SMBTransactionCreate {
pub fn new(filename: Vec<u8>, disp: u32, del: bool, dir: bool) -> Self {
return Self {
disposition: disp,
delete_on_close: del,
directory: dir,
filename: filename,
..Default::default()
}
}
}
#[derive(Default, Debug)]
pub struct SMBTransactionNegotiate {
pub smb_ver: u8,
pub dialects: Vec<Vec<u8>>,
pub dialects2: Vec<Vec<u8>>,
// SMB1 doesn't have the client GUID
pub client_guid: Option<Vec<u8>>,
pub server_guid: Vec<u8>,
}
impl SMBTransactionNegotiate {
pub fn new(smb_ver: u8) -> Self {
return Self {
smb_ver: smb_ver,
server_guid: Vec::with_capacity(16),
..Default::default()
}
}
}
#[derive(Default, Debug)]
pub struct SMBTransactionTreeConnect {
pub is_pipe: bool,
pub share_type: u8,
pub tree_id: u32,
pub share_name: Vec<u8>,
/// SMB1 service strings
pub req_service: Option<Vec<u8>>,
pub res_service: Option<Vec<u8>>,
}
impl SMBTransactionTreeConnect {
pub fn new(share_name: Vec<u8>) -> Self {
return Self {
share_name:share_name,
..Default::default()
}
}
}
#[derive(Debug)]
pub struct SMBTransaction {
pub id: u64, /// internal id
/// version, command and status
pub vercmd: SMBVerCmdStat,
/// session id, tree id, etc.
pub hdr: SMBCommonHdr,
/// for state tracking. false means this side is in progress, true
/// that it's complete.
pub request_done: bool,
pub response_done: bool,
/// Command specific data
pub type_data: Option<SMBTransactionTypeData>,
pub de_state: Option<*mut DetectEngineState>,
pub events: *mut AppLayerDecoderEvents,
pub tx_data: AppLayerTxData,
}
impl SMBTransaction {
pub fn new() -> Self {
return Self {
id: 0,
vercmd: SMBVerCmdStat::new(),
hdr: SMBCommonHdr::init(),
request_done: false,
response_done: false,
type_data: None,
de_state: None,
events: std::ptr::null_mut(),
tx_data: AppLayerTxData::new(),
}
}
pub fn set_status(&mut self, status: u32, is_dos_error: bool)
{
if is_dos_error {
self.vercmd.set_status_dos_error(status);
} else {
self.vercmd.set_ntstatus(status);
}
}
pub fn free(&mut self) {
if self.events != std::ptr::null_mut() {
sc_app_layer_decoder_events_free_events(&mut self.events);
}
match self.de_state {
Some(state) => {
sc_detect_engine_state_free(state);
}
_ => {}
}
}
}
impl Drop for SMBTransaction {
fn drop(&mut self) {
self.free();
}
}
#[derive(Hash, Eq, PartialEq, Debug, Clone)]
pub struct SMBFileGUIDOffset {
pub guid: Vec<u8>,
pub offset: u64,
}
impl SMBFileGUIDOffset {
pub fn new(guid: Vec<u8>, offset: u64) -> Self {
Self {
guid:guid,
offset:offset,
}
}
}
/// type values to make sure we're not mixing things
/// up in hashmap lookups
pub const SMBHDR_TYPE_GUID: u32 = 1;
pub const SMBHDR_TYPE_SHARE: u32 = 2;
pub const SMBHDR_TYPE_FILENAME: u32 = 3;
pub const SMBHDR_TYPE_OFFSET: u32 = 4;
pub const SMBHDR_TYPE_GENERICTX: u32 = 5;
pub const SMBHDR_TYPE_HEADER: u32 = 6;
pub const SMBHDR_TYPE_MAX_SIZE: u32 = 7; // max resp size for SMB1_COMMAND_TRANS
pub const SMBHDR_TYPE_TRANS_FRAG: u32 = 8;
pub const SMBHDR_TYPE_TREE: u32 = 9;
pub const SMBHDR_TYPE_DCERPCTX: u32 = 10;
#[derive(Default, Hash, Eq, PartialEq, Debug)]
pub struct SMBCommonHdr {
pub ssn_id: u64,
pub tree_id: u32,
pub rec_type: u32,
pub msg_id: u64,
}
impl SMBCommonHdr {
pub fn init() -> Self {
Default::default()
}
pub fn new(rec_type: u32, ssn_id: u64, tree_id: u32, msg_id: u64) -> Self {
Self {
rec_type : rec_type,
ssn_id : ssn_id,
tree_id : tree_id,
msg_id : msg_id,
}
}
pub fn from2(r: &Smb2Record, rec_type: u32) -> SMBCommonHdr {
let tree_id = match rec_type {
SMBHDR_TYPE_TREE => { 0 },
_ => r.tree_id,
};
let msg_id = match rec_type {
SMBHDR_TYPE_TRANS_FRAG | SMBHDR_TYPE_SHARE => { 0 },
_ => { r.message_id as u64 },
};
SMBCommonHdr {
rec_type : rec_type,
ssn_id : r.session_id,
tree_id : tree_id,
msg_id : msg_id,
}
}
pub fn from1(r: &SmbRecord, rec_type: u32) -> SMBCommonHdr {
let tree_id = match rec_type {
SMBHDR_TYPE_TREE => { 0 },
_ => r.tree_id as u32,
};
let msg_id = match rec_type {
SMBHDR_TYPE_TRANS_FRAG | SMBHDR_TYPE_SHARE => { 0 },
_ => { r.multiplex_id as u64 },
};
SMBCommonHdr {
rec_type : rec_type,
ssn_id : r.ssn_id as u64,
tree_id : tree_id,
msg_id : msg_id,
}
}
// don't include tree id
pub fn compare(&self, hdr: &SMBCommonHdr) -> bool {
self.rec_type == hdr.rec_type && self.ssn_id == hdr.ssn_id &&
self.msg_id == hdr.msg_id
}
}
#[derive(Hash, Eq, PartialEq, Debug)]
pub struct SMBHashKeyHdrGuid {
hdr: SMBCommonHdr,
guid: Vec<u8>,
}
impl SMBHashKeyHdrGuid {
pub fn new(hdr: SMBCommonHdr, guid: Vec<u8>) -> Self {
Self {
hdr: hdr, guid: guid,
}
}
}
#[derive(Hash, Eq, PartialEq, Debug)]
pub struct SMBTree {
pub name: Vec<u8>,
pub is_pipe: bool,
}
impl SMBTree {
pub fn new(name: Vec<u8>, is_pipe: bool) -> Self {
Self {
name:name,
is_pipe:is_pipe,
}
}
}
pub fn u32_as_bytes(i: u32) -> [u8;4] {
let o1: u8 = ((i >> 24) & 0xff) as u8;
let o2: u8 = ((i >> 16) & 0xff) as u8;
let o3: u8 = ((i >> 8) & 0xff) as u8;
let o4: u8 = (i & 0xff) as u8;
return [o1, o2, o3, o4]
}
pub struct SMBState<> {
/// map ssn/tree/msgid to vec (guid/name/share)
pub ssn2vec_map: HashMap<SMBCommonHdr, Vec<u8>>,
/// map guid to filename
pub guid2name_map: HashMap<Vec<u8>, Vec<u8>>,
/// map ssn key to read offset
pub ssn2vecoffset_map: HashMap<SMBCommonHdr, SMBFileGUIDOffset>,
pub ssn2tree_map: HashMap<SMBCommonHdr, SMBTree>,
// store partial data records that are transfered in multiple
// requests for DCERPC.
pub ssnguid2vec_map: HashMap<SMBHashKeyHdrGuid, Vec<u8>>,
pub files: Files,
skip_ts: u32,
skip_tc: u32,
pub file_ts_left : u32,
pub file_tc_left : u32,
pub file_ts_guid : Vec<u8>,
pub file_tc_guid : Vec<u8>,
pub ts_ssn_gap: bool,
pub tc_ssn_gap: bool,
pub ts_gap: bool, // last TS update was gap
pub tc_gap: bool, // last TC update was gap
pub ts_trunc: bool, // no more data for TOSERVER
pub tc_trunc: bool, // no more data for TOCLIENT
/// true as long as we have file txs that are in a post-gap
/// state. It means we'll do extra house keeping for those.
check_post_gap_file_txs: bool,
post_gap_files_checked: bool,
/// transactions list
pub transactions: Vec<SMBTransaction>,
/// tx counter for assigning incrementing id's to tx's
tx_id: u64,
/// SMB2 dialect or 0 if not set or SMB1
pub dialect: u16,
/// contains name of SMB1 dialect
pub dialect_vec: Option<Vec<u8>>, // used if dialect == 0
/// dcerpc interfaces, stored here to be able to match
/// them while inspecting DCERPC REQUEST txs
pub dcerpc_ifaces: Option<Vec<DCERPCIface>>,
/// Timestamp in seconds of last update. This is packet time,
/// potentially coming from pcaps.
ts: u64,
}
impl SMBState {
/// Allocation function for a new TLS parser instance
pub fn new() -> Self {
Self {
ssn2vec_map:HashMap::new(),
guid2name_map:HashMap::new(),
ssn2vecoffset_map:HashMap::new(),
ssn2tree_map:HashMap::new(),
ssnguid2vec_map:HashMap::new(),
files: Files::default(),
skip_ts:0,
skip_tc:0,
file_ts_left:0,
file_tc_left:0,
file_ts_guid:Vec::new(),
file_tc_guid:Vec::new(),
ts_ssn_gap: false,
tc_ssn_gap: false,
ts_gap: false,
tc_gap: false,
ts_trunc: false,
tc_trunc: false,
check_post_gap_file_txs: false,
post_gap_files_checked: false,
transactions: Vec::new(),
tx_id:0,
dialect:0,
dialect_vec: None,
dcerpc_ifaces: None,
ts: 0,
}
}
pub fn free(&mut self) {
//self._debug_state_stats();
self._debug_tx_stats();
}
pub fn new_tx(&mut self) -> SMBTransaction {
let mut tx = SMBTransaction::new();
self.tx_id += 1;
tx.id = self.tx_id;
SCLogDebug!("TX {} created", tx.id);
return tx;
}
pub fn free_tx(&mut self, tx_id: u64) {
SCLogDebug!("Freeing TX with ID {} TX.ID {}", tx_id, tx_id+1);
let len = self.transactions.len();
let mut found = false;
let mut index = 0;
for i in 0..len {
let tx = &self.transactions[i];
if tx.id == tx_id + 1 {
found = true;
index = i;
SCLogDebug!("tx {} progress {}/{}", tx.id, tx.request_done, tx.response_done);
break;
}
}
if found {
SCLogDebug!("freeing TX with ID {} TX.ID {} at index {} left: {} max id: {}",
tx_id, tx_id+1, index, self.transactions.len(), self.tx_id);
self.transactions.remove(index);
}
}
// for use with the C API call StateGetTxIterator
pub fn get_tx_iterator(&mut self, min_tx_id: u64, state: &mut u64) ->
Option<(&SMBTransaction, u64, bool)>
{
let mut index = *state as usize;
let len = self.transactions.len();
// find tx that is >= min_tx_id
while index < len {
let tx = &self.transactions[index];
if tx.id < min_tx_id + 1 {
index += 1;
continue;
}
// store current index in the state and not the next
// as transactions might be freed between now and the
// next time we are called.
*state = index as u64;
//SCLogDebug!("returning tx_id {} has_next? {} (len {} index {}), tx {:?}",
// tx.id - 1, (len - index) > 1, len, index, tx);
return Some((tx, tx.id - 1, (len - index) > 1));
}
return None;
}
pub fn get_tx_by_id(&mut self, tx_id: u64) -> Option<&SMBTransaction> {
/*
if self.transactions.len() > 100 {
SCLogNotice!("get_tx_by_id: tx_id={} in list={}", tx_id, self.transactions.len());
self._dump_txs();
panic!("txs exploded");
}
*/
for tx in &mut self.transactions {
if tx.id == tx_id + 1 {
let ver = tx.vercmd.get_version();
let mut _smbcmd;
if ver == 2 {
let (_, cmd) = tx.vercmd.get_smb2_cmd();
_smbcmd = cmd;
} else {
let (_, cmd) = tx.vercmd.get_smb1_cmd();
_smbcmd = cmd as u16;
}
SCLogDebug!("Found SMB TX: id {} ver:{} cmd:{} progress {}/{} type_data {:?}",
tx.id, ver, _smbcmd, tx.request_done, tx.response_done, tx.type_data);
return Some(tx);
}
}
SCLogDebug!("Failed to find SMB TX with ID {}", tx_id);
return None;
}
fn update_ts(&mut self, ts: u64) {
if ts != self.ts {
self.ts = ts;
self.post_gap_files_checked = false;
}
}
/* generic TX has no type_data and is only used to
* track a single cmd request/reply pair. */
pub fn new_generic_tx(&mut self, smb_ver: u8, smb_cmd: u16, key: SMBCommonHdr)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
if smb_ver == 1 && smb_cmd <= 255 {
tx.vercmd.set_smb1_cmd(smb_cmd as u8);
} else if smb_ver == 2 {
tx.vercmd.set_smb2_cmd(smb_cmd);
}
tx.type_data = None;
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
tx.hdr = key;
SCLogDebug!("SMB: TX GENERIC created: ID {} tx list {} {:?}",
tx.id, self.transactions.len(), &tx);
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
pub fn get_last_tx(&mut self, smb_ver: u8, smb_cmd: u16)
-> Option<&mut SMBTransaction>
{
let tx_ref = self.transactions.last_mut();
match tx_ref {
Some(tx) => {
let found = if tx.vercmd.get_version() == smb_ver {
if smb_ver == 1 {
let (_, cmd) = tx.vercmd.get_smb1_cmd();
cmd as u16 == smb_cmd
} else if smb_ver == 2 {
let (_, cmd) = tx.vercmd.get_smb2_cmd();
cmd == smb_cmd
} else {
false
}
} else {
false
};
if found {
return Some(tx);
}
},
None => { },
}
return None;
}
pub fn get_generic_tx(&mut self, smb_ver: u8, smb_cmd: u16, key: &SMBCommonHdr)
-> Option<&mut SMBTransaction>
{
for tx in &mut self.transactions {
let found = if tx.vercmd.get_version() == smb_ver {
if smb_ver == 1 {
let (_, cmd) = tx.vercmd.get_smb1_cmd();
cmd as u16 == smb_cmd && tx.hdr.compare(key)
} else if smb_ver == 2 {
let (_, cmd) = tx.vercmd.get_smb2_cmd();
cmd == smb_cmd && tx.hdr.compare(key)
} else {
false
}
} else {
false
};
if found {
return Some(tx);
}
}
return None;
}
pub fn new_negotiate_tx(&mut self, smb_ver: u8)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
if smb_ver == 1 {
tx.vercmd.set_smb1_cmd(SMB1_COMMAND_NEGOTIATE_PROTOCOL);
} else if smb_ver == 2 {
tx.vercmd.set_smb2_cmd(SMB2_COMMAND_NEGOTIATE_PROTOCOL);
}
tx.type_data = Some(SMBTransactionTypeData::NEGOTIATE(
SMBTransactionNegotiate::new(smb_ver)));
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
SCLogDebug!("SMB: TX NEGOTIATE created: ID {} SMB ver {}", tx.id, smb_ver);
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
pub fn get_negotiate_tx(&mut self, smb_ver: u8)
-> Option<&mut SMBTransaction>
{
for tx in &mut self.transactions {
let found = match tx.type_data {
Some(SMBTransactionTypeData::NEGOTIATE(ref x)) => {
if x.smb_ver == smb_ver {
true
} else {
false
}
},
_ => { false },
};
if found {
return Some(tx);
}
}
return None;
}
pub fn new_treeconnect_tx(&mut self, hdr: SMBCommonHdr, name: Vec<u8>)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
tx.hdr = hdr;
tx.type_data = Some(SMBTransactionTypeData::TREECONNECT(
SMBTransactionTreeConnect::new(name.to_vec())));
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
SCLogDebug!("SMB: TX TREECONNECT created: ID {} NAME {}",
tx.id, String::from_utf8_lossy(&name));
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
pub fn get_treeconnect_tx(&mut self, hdr: SMBCommonHdr)
-> Option<&mut SMBTransaction>
{
for tx in &mut self.transactions {
let hit = tx.hdr.compare(&hdr) && match tx.type_data {
Some(SMBTransactionTypeData::TREECONNECT(_)) => { true },
_ => { false },
};
if hit {
return Some(tx);
}
}
return None;
}
pub fn new_create_tx(&mut self, file_name: &Vec<u8>,
disposition: u32, del: bool, dir: bool,
hdr: SMBCommonHdr)
-> &mut SMBTransaction
{
let mut tx = self.new_tx();
tx.hdr = hdr;
tx.type_data = Some(SMBTransactionTypeData::CREATE(
SMBTransactionCreate::new(
file_name.to_vec(), disposition,
del, dir)));
tx.request_done = true;
tx.response_done = self.tc_trunc; // no response expected if tc is truncated
self.transactions.push(tx);
let tx_ref = self.transactions.last_mut();
return tx_ref.unwrap();
}
pub fn get_create_tx_by_hdr(&mut self, hdr: &SMBCommonHdr)
-> Option<&mut SMBTransaction>
{
for tx in &mut self.transactions {
let found = match tx.type_data {
Some(SMBTransactionTypeData::CREATE(ref _d)) => {
tx.hdr.compare(&hdr)
},
_ => { false },
};
if found {
SCLogDebug!("SMB: Found SMB create TX with ID {}", tx.id);
return Some(tx);
}
}
SCLogDebug!("SMB: Failed to find SMB create TX with key {:?}", hdr);
return None;
}
pub fn get_service_for_guid(&self, guid: &[u8]) -> (&'static str, bool)
{
let (name, is_dcerpc) = match self.guid2name_map.get(&guid.to_vec()) {
Some(n) => {
let mut s = n.as_slice();
// skip leading \ if we have it
if s.len() > 1 && s[0] == 0x5c_u8 {
s = &s[1..];
}
match str::from_utf8(s) {
Ok("PSEXESVC") => ("PSEXESVC", false),
Ok("svcctl") => ("svcctl", true),
Ok("srvsvc") => ("srvsvc", true),
Ok("atsvc") => ("atsvc", true),
Ok("lsarpc") => ("lsarpc", true),
Ok("samr") => ("samr", true),
Ok("spoolss") => ("spoolss", true),
Ok("winreg") => ("winreg", true),
Ok("suricata::dcerpc") => ("unknown", true),
Err(_) => ("MALFORMED", false),
Ok(&_) => {
SCLogDebug!("don't know {}", String::from_utf8_lossy(&n));
("UNKNOWN", false)
},
}
},
_ => { ("UNKNOWN", false) },
};
SCLogDebug!("service {} is_dcerpc {}", name, is_dcerpc);
(&name, is_dcerpc)
}
fn post_gap_housekeeping_for_files(&mut self)
{
let mut post_gap_txs = false;
for tx in &mut self.transactions {
if let Some(SMBTransactionTypeData::FILE(ref mut f)) = tx.type_data {
if f.post_gap_ts > 0 {
if self.ts > f.post_gap_ts {
tx.request_done = true;
tx.response_done = true;
let (files, flags) = self.files.get(f.direction);
f.file_tracker.trunc(files, flags);
} else {
post_gap_txs = true;
}
}
}
}
self.check_post_gap_file_txs = post_gap_txs;
}
/* after a gap we will consider all transactions complete for our
* direction. File transfer transactions are an exception. Those
* can handle gaps. For the file transactions we set the current
* (flow) time and prune them in 60 seconds if no update for them
* was received. */
fn post_gap_housekeeping(&mut self, dir: u8)
{
if self.ts_ssn_gap && dir == STREAM_TOSERVER {
for tx in &mut self.transactions {
if tx.id >= self.tx_id {
SCLogDebug!("post_gap_housekeeping: done");
break;
}
if let Some(SMBTransactionTypeData::FILE(ref mut f)) = tx.type_data {
// leaving FILE txs open as they can deal with gaps. We
// remove them after 60 seconds of no activity though.
if f.post_gap_ts == 0 {
f.post_gap_ts = self.ts + 60;
self.check_post_gap_file_txs = true;
}
} else {
SCLogDebug!("post_gap_housekeeping: tx {} marked as done TS", tx.id);
tx.request_done = true;
}
}
} else if self.tc_ssn_gap && dir == STREAM_TOCLIENT {
for tx in &mut self.transactions {
if tx.id >= self.tx_id {
SCLogDebug!("post_gap_housekeeping: done");
break;
}
if let Some(SMBTransactionTypeData::FILE(ref mut f)) = tx.type_data {
// leaving FILE txs open as they can deal with gaps. We
// remove them after 60 seconds of no activity though.
if f.post_gap_ts == 0 {
f.post_gap_ts = self.ts + 60;
self.check_post_gap_file_txs = true;
}
} else {
SCLogDebug!("post_gap_housekeeping: tx {} marked as done TC", tx.id);
tx.request_done = true;
tx.response_done = true;
}
}
}
}
pub fn set_file_left(&mut self, direction: u8, rec_size: u32, data_size: u32, fuid: Vec<u8>)
{
let left = rec_size.saturating_sub(data_size);
if direction == STREAM_TOSERVER {
self.file_ts_left = left;
self.file_ts_guid = fuid;
} else {
self.file_tc_left = left;
self.file_tc_guid = fuid;
}
}
pub fn set_skip(&mut self, direction: u8, rec_size: u32, data_size: u32)
{
let skip = rec_size.saturating_sub(data_size);
if direction == STREAM_TOSERVER {
self.skip_ts = skip;
} else {
self.skip_tc = skip;
}
}
// return how much data we consumed
fn handle_skip(&mut self, direction: u8, input_size: u32) -> u32 {
let mut skip_left = if direction == STREAM_TOSERVER {
self.skip_ts
} else {
self.skip_tc
};
if skip_left == 0 {
return 0
}
SCLogDebug!("skip_left {} input_size {}", skip_left, input_size);
let consumed = if skip_left >= input_size {
input_size
} else {
skip_left
};
if skip_left <= input_size {
skip_left = 0;
} else {
skip_left -= input_size;
}
if direction == STREAM_TOSERVER {
self.skip_ts = skip_left;
} else {
self.skip_tc = skip_left;
}
return consumed;
}
/// return bytes consumed
pub fn parse_tcp_data_ts_partial<'b>(&mut self, input: &'b[u8]) -> usize
{
SCLogDebug!("incomplete of size {}", input.len());
if input.len() < 512 {
// check for malformed data. Wireshark reports as
// 'NBSS continuation data'. If it's invalid we're
// lost so we give up.
if input.len() > 8 {
match parse_nbss_record_partial(input) {
Ok((_, ref hdr)) => {
if !hdr.is_smb() {
SCLogDebug!("partial NBSS, not SMB and no known msg type {}", hdr.message_type);
self.trunc_ts();
return 0;
}
},
_ => {},
}
}
return 0;
}
match parse_nbss_record_partial(input) {
Ok((output, ref nbss_part_hdr)) => {
SCLogDebug!("parse_nbss_record_partial ok, output len {}", output.len());
if nbss_part_hdr.message_type == NBSS_MSGTYPE_SESSION_MESSAGE {
match parse_smb_version(&nbss_part_hdr.data) {
Ok((_, ref smb)) => {
SCLogDebug!("SMB {:?}", smb);
if smb.version == 0xff_u8 { // SMB1
SCLogDebug!("SMBv1 record");
match parse_smb_record(&nbss_part_hdr.data) {
Ok((_, ref r)) => {
if r.command == SMB1_COMMAND_WRITE_ANDX {
// see if it's a write to a pipe. We only handle those
// if complete.
let tree_key = SMBCommonHdr::new(SMBHDR_TYPE_SHARE,
r.ssn_id as u64, r.tree_id as u32, 0);
let is_pipe = match self.ssn2tree_map.get(&tree_key) {
Some(n) => n.is_pipe,
None => false,
};
if is_pipe {
return 0;
}
smb1_write_request_record(self, r, SMB1_HEADER_SIZE, SMB1_COMMAND_WRITE_ANDX);
let consumed = input.len() - output.len();
return consumed;
}
},
_ => { },
}
} else if smb.version == 0xfe_u8 { // SMB2
SCLogDebug!("NBSS record {:?}", nbss_part_hdr);
SCLogDebug!("SMBv2 record");
match parse_smb2_request_record(&nbss_part_hdr.data) {
Ok((_, ref smb_record)) => {
SCLogDebug!("SMB2: partial record {}",
&smb2_command_string(smb_record.command));
if smb_record.command == SMB2_COMMAND_WRITE {
smb2_write_request_record(self, smb_record);
let consumed = input.len() - output.len();
SCLogDebug!("consumed {}", consumed);
return consumed;
}
},
_ => { },
}
}
// no SMB3 here yet, will buffer full records
},
_ => { },
}
}
},
_ => { },
}
return 0;
}
/// Parsing function, handling TCP chunks fragmentation
pub fn parse_tcp_data_ts<'b>(&mut self, i: &'b[u8]) -> AppLayerResult
{
let mut cur_i = i;
let consumed = self.handle_skip(STREAM_TOSERVER, cur_i.len() as u32);
if consumed > 0 {
if consumed > cur_i.len() as u32 {
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
cur_i = &cur_i[consumed as usize..];
}
// take care of in progress file chunk transfers
// and skip buffer beyond it
let consumed = self.filetracker_update(STREAM_TOSERVER, cur_i, 0);
if consumed > 0 {
if consumed > cur_i.len() as u32 {
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
cur_i = &cur_i[consumed as usize..];
}
if cur_i.len() == 0 {
return AppLayerResult::ok();
}
// gap
if self.ts_gap {
SCLogDebug!("TS trying to catch up after GAP (input {})", cur_i.len());
while cur_i.len() > 0 { // min record size
match search_smb_record(cur_i) {
Ok((_, pg)) => {
SCLogDebug!("smb record found");
let smb2_offset = cur_i.len() - pg.len();
if smb2_offset < 4 {
cur_i = &cur_i[smb2_offset+4..];
continue; // see if we have another record in our data
}
let nbss_offset = smb2_offset - 4;
cur_i = &cur_i[nbss_offset..];
self.ts_gap = false;
break;
},
_ => {
let mut consumed = i.len();
if consumed < 4 {
consumed = 0;
} else {
consumed = consumed - 3;
}
SCLogDebug!("smb record NOT found");
return AppLayerResult::incomplete(consumed as u32, 8);
},
}
}
}
while cur_i.len() > 0 { // min record size
match parse_nbss_record(cur_i) {
Ok((rem, ref nbss_hdr)) => {
if nbss_hdr.message_type == NBSS_MSGTYPE_SESSION_MESSAGE {
// we have the full records size worth of data,
// let's parse it
match parse_smb_version(&nbss_hdr.data) {
Ok((_, ref smb)) => {
SCLogDebug!("SMB {:?}", smb);
if smb.version == 0xff_u8 { // SMB1
SCLogDebug!("SMBv1 record");
match parse_smb_record(&nbss_hdr.data) {
Ok((_, ref smb_record)) => {
smb1_request_record(self, smb_record);
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
} else if smb.version == 0xfe_u8 { // SMB2
let mut nbss_data = nbss_hdr.data;
while nbss_data.len() > 0 {
SCLogDebug!("SMBv2 record");
match parse_smb2_request_record(&nbss_data) {
Ok((nbss_data_rem, ref smb_record)) => {
SCLogDebug!("nbss_data_rem {}", nbss_data_rem.len());
smb2_request_record(self, smb_record);
nbss_data = nbss_data_rem;
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
}
} else if smb.version == 0xfd_u8 { // SMB3 transform
let mut nbss_data = nbss_hdr.data;
while nbss_data.len() > 0 {
SCLogDebug!("SMBv3 transform record");
match parse_smb3_transform_record(&nbss_data) {
Ok((nbss_data_rem, ref _smb3_record)) => {
nbss_data = nbss_data_rem;
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
}
}
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
} else {
SCLogDebug!("NBSS message {:X}", nbss_hdr.message_type);
}
cur_i = rem;
},
Err(nom::Err::Incomplete(needed)) => {
if let nom::Needed::Size(n) = needed {
// 512 is the minimum for parse_tcp_data_ts_partial
if n >= 512 && cur_i.len() < 512 {
let total_consumed = i.len() - cur_i.len();
return AppLayerResult::incomplete(total_consumed as u32, 512);
}
let consumed = self.parse_tcp_data_ts_partial(cur_i);
if consumed == 0 {
// if we consumed none we will buffer the entire record
let total_consumed = i.len() - cur_i.len();
SCLogDebug!("setting consumed {} need {} needed {:?} total input {}",
total_consumed, n, needed, i.len());
let need = n + 4; // Incomplete returns size of data minus NBSS header
return AppLayerResult::incomplete(total_consumed as u32, need as u32);
}
// tracking a write record, which we don't need to
// queue up at the stream level, but can feed to us
// in small chunks
return AppLayerResult::ok();
} else {
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
},
Err(_) => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
};
self.post_gap_housekeeping(STREAM_TOSERVER);
if self.check_post_gap_file_txs && !self.post_gap_files_checked {
self.post_gap_housekeeping_for_files();
self.post_gap_files_checked = true;
}
AppLayerResult::ok()
}
/// return bytes consumed
pub fn parse_tcp_data_tc_partial<'b>(&mut self, input: &'b[u8]) -> usize
{
SCLogDebug!("incomplete of size {}", input.len());
if input.len() < 512 {
// check for malformed data. Wireshark reports as
// 'NBSS continuation data'. If it's invalid we're
// lost so we give up.
if input.len() > 8 {
match parse_nbss_record_partial(input) {
Ok((_, ref hdr)) => {
if !hdr.is_smb() {
SCLogDebug!("partial NBSS, not SMB and no known msg type {}", hdr.message_type);
self.trunc_tc();
return 0;
}
},
_ => {},
}
}
return 0;
}
match parse_nbss_record_partial(input) {
Ok((output, ref nbss_part_hdr)) => {
SCLogDebug!("parse_nbss_record_partial ok, output len {}", output.len());
if nbss_part_hdr.message_type == NBSS_MSGTYPE_SESSION_MESSAGE {
match parse_smb_version(&nbss_part_hdr.data) {
Ok((_, ref smb)) => {
SCLogDebug!("SMB {:?}", smb);
if smb.version == 255u8 { // SMB1
SCLogDebug!("SMBv1 record");
match parse_smb_record(&nbss_part_hdr.data) {
Ok((_, ref r)) => {
SCLogDebug!("SMB1: partial record {}",
r.command);
if r.command == SMB1_COMMAND_READ_ANDX {
let tree_key = SMBCommonHdr::new(SMBHDR_TYPE_SHARE,
r.ssn_id as u64, r.tree_id as u32, 0);
let is_pipe = match self.ssn2tree_map.get(&tree_key) {
Some(n) => n.is_pipe,
None => false,
};
if is_pipe {
return 0;
}
smb1_read_response_record(self, r, SMB1_HEADER_SIZE);
let consumed = input.len() - output.len();
return consumed;
}
},
_ => { },
}
} else if smb.version == 254u8 { // SMB2
SCLogDebug!("SMBv2 record");
match parse_smb2_response_record(&nbss_part_hdr.data) {
Ok((_, ref smb_record)) => {
SCLogDebug!("SMB2: partial record {}",
&smb2_command_string(smb_record.command));
if smb_record.command == SMB2_COMMAND_READ {
smb2_read_response_record(self, smb_record);
let consumed = input.len() - output.len();
return consumed;
}
},
_ => { },
}
}
// no SMB3 here yet, will buffer full records
},
_ => { },
}
}
},
_ => { },
}
return 0;
}
/// Parsing function, handling TCP chunks fragmentation
pub fn parse_tcp_data_tc<'b>(&mut self, i: &'b[u8]) -> AppLayerResult
{
let mut cur_i = i;
let consumed = self.handle_skip(STREAM_TOCLIENT, cur_i.len() as u32);
if consumed > 0 {
if consumed > cur_i.len() as u32 {
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
cur_i = &cur_i[consumed as usize..];
}
// take care of in progress file chunk transfers
// and skip buffer beyond it
let consumed = self.filetracker_update(STREAM_TOCLIENT, cur_i, 0);
if consumed > 0 {
if consumed > cur_i.len() as u32 {
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
cur_i = &cur_i[consumed as usize..];
}
if cur_i.len() == 0 {
return AppLayerResult::ok();
}
// gap
if self.tc_gap {
SCLogDebug!("TC trying to catch up after GAP (input {})", cur_i.len());
while cur_i.len() > 0 { // min record size
match search_smb_record(cur_i) {
Ok((_, pg)) => {
SCLogDebug!("smb record found");
let smb2_offset = cur_i.len() - pg.len();
if smb2_offset < 4 {
cur_i = &cur_i[smb2_offset+4..];
continue; // see if we have another record in our data
}
let nbss_offset = smb2_offset - 4;
cur_i = &cur_i[nbss_offset..];
self.tc_gap = false;
break;
},
_ => {
let mut consumed = i.len();
if consumed < 4 {
consumed = 0;
} else {
consumed = consumed - 3;
}
SCLogDebug!("smb record NOT found");
return AppLayerResult::incomplete(consumed as u32, 8);
},
}
}
}
while cur_i.len() > 0 { // min record size
match parse_nbss_record(cur_i) {
Ok((rem, ref nbss_hdr)) => {
if nbss_hdr.message_type == NBSS_MSGTYPE_SESSION_MESSAGE {
// we have the full records size worth of data,
// let's parse it
match parse_smb_version(&nbss_hdr.data) {
Ok((_, ref smb)) => {
SCLogDebug!("SMB {:?}", smb);
if smb.version == 0xff_u8 { // SMB1
SCLogDebug!("SMBv1 record");
match parse_smb_record(&nbss_hdr.data) {
Ok((_, ref smb_record)) => {
smb1_response_record(self, smb_record);
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
} else if smb.version == 0xfe_u8 { // SMB2
let mut nbss_data = nbss_hdr.data;
while nbss_data.len() > 0 {
SCLogDebug!("SMBv2 record");
match parse_smb2_response_record(&nbss_data) {
Ok((nbss_data_rem, ref smb_record)) => {
smb2_response_record(self, smb_record);
nbss_data = nbss_data_rem;
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
}
} else if smb.version == 0xfd_u8 { // SMB3 transform
let mut nbss_data = nbss_hdr.data;
while nbss_data.len() > 0 {
SCLogDebug!("SMBv3 transform record");
match parse_smb3_transform_record(&nbss_data) {
Ok((nbss_data_rem, ref _smb3_record)) => {
nbss_data = nbss_data_rem;
},
_ => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
}
}
},
Err(nom::Err::Incomplete(_)) => {
// not enough data to contain basic SMB hdr
// TODO event: empty NBSS_MSGTYPE_SESSION_MESSAGE
},
Err(_) => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
} else {
SCLogDebug!("NBSS message {:X}", nbss_hdr.message_type);
}
cur_i = rem;
},
Err(nom::Err::Incomplete(needed)) => {
SCLogDebug!("INCOMPLETE have {} needed {:?}", cur_i.len(), needed);
if let nom::Needed::Size(n) = needed {
// 512 is the minimum for parse_tcp_data_tc_partial
if n >= 512 && cur_i.len() < 512 {
let total_consumed = i.len() - cur_i.len();
return AppLayerResult::incomplete(total_consumed as u32, 512);
}
let consumed = self.parse_tcp_data_tc_partial(cur_i);
if consumed == 0 {
// if we consumed none we will buffer the entire record
let total_consumed = i.len() - cur_i.len();
SCLogDebug!("setting consumed {} need {} needed {:?} total input {}",
total_consumed, n, needed, i.len());
let need = n + 4; // Incomplete returns size of data minus NBSS header
return AppLayerResult::incomplete(total_consumed as u32, need as u32);
}
// tracking a read record, which we don't need to
// queue up at the stream level, but can feed to us
// in small chunks
return AppLayerResult::ok();
} else {
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
},
Err(_) => {
self.set_event(SMBEvent::MalformedData);
return AppLayerResult::err();
},
}
};
self.post_gap_housekeeping(STREAM_TOCLIENT);
if self.check_post_gap_file_txs && !self.post_gap_files_checked {
self.post_gap_housekeeping_for_files();
self.post_gap_files_checked = true;
}
self._debug_tx_stats();
AppLayerResult::ok()
}
/// handle a gap in the TOSERVER direction
/// returns: 0 ok, 1 unrecoverable error
pub fn parse_tcp_data_ts_gap(&mut self, gap_size: u32) -> AppLayerResult {
let consumed = self.handle_skip(STREAM_TOSERVER, gap_size);
if consumed < gap_size {
let new_gap_size = gap_size - consumed;
let gap = vec![0; new_gap_size as usize];
let consumed2 = self.filetracker_update(STREAM_TOSERVER, &gap, new_gap_size);
if consumed2 > new_gap_size {
SCLogDebug!("consumed more than GAP size: {} > {}", consumed2, new_gap_size);
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
}
SCLogDebug!("GAP of size {} in toserver direction", gap_size);
self.ts_ssn_gap = true;
self.ts_gap = true;
return AppLayerResult::ok();
}
/// handle a gap in the TOCLIENT direction
/// returns: 0 ok, 1 unrecoverable error
pub fn parse_tcp_data_tc_gap(&mut self, gap_size: u32) -> AppLayerResult {
let consumed = self.handle_skip(STREAM_TOCLIENT, gap_size);
if consumed < gap_size {
let new_gap_size = gap_size - consumed;
let gap = vec![0; new_gap_size as usize];
let consumed2 = self.filetracker_update(STREAM_TOCLIENT, &gap, new_gap_size);
if consumed2 > new_gap_size {
SCLogDebug!("consumed more than GAP size: {} > {}", consumed2, new_gap_size);
self.set_event(SMBEvent::InternalError);
return AppLayerResult::err();
}
}
SCLogDebug!("GAP of size {} in toclient direction", gap_size);
self.tc_ssn_gap = true;
self.tc_gap = true;
return AppLayerResult::ok();
}
pub fn trunc_ts(&mut self) {
SCLogDebug!("TRUNC TS");
self.ts_trunc = true;
for tx in &mut self.transactions {
if !tx.request_done {
SCLogDebug!("TRUNCING TX {} in TOSERVER direction", tx.id);
tx.request_done = true;
}
}
}
pub fn trunc_tc(&mut self) {
SCLogDebug!("TRUNC TC");
self.tc_trunc = true;
for tx in &mut self.transactions {
if !tx.response_done {
SCLogDebug!("TRUNCING TX {} in TOCLIENT direction", tx.id);
tx.response_done = true;
}
}
}
}
/// Returns *mut SMBState
#[no_mangle]
pub extern "C" fn rs_smb_state_new(_orig_state: *mut std::os::raw::c_void, _orig_proto: AppProto) -> *mut std::os::raw::c_void {
let state = SMBState::new();
let boxed = Box::new(state);
SCLogDebug!("allocating state");
return unsafe{transmute(boxed)};
}
/// Params:
/// - state: *mut SMBState as void pointer
#[no_mangle]
pub extern "C" fn rs_smb_state_free(state: *mut std::os::raw::c_void) {
// Just unbox...
SCLogDebug!("freeing state");
let mut smb_state: Box<SMBState> = unsafe{transmute(state)};
smb_state.free();
}
/// C binding parse a SMB request. Returns 1 on success, -1 on failure.
#[no_mangle]
pub extern "C" fn rs_smb_parse_request_tcp(flow: *const Flow,
state: *mut ffi::c_void,
_pstate: *mut std::os::raw::c_void,
input: *const u8,
input_len: u32,
_data: *const std::os::raw::c_void,
flags: u8)
-> AppLayerResult
{
let buf = unsafe{std::slice::from_raw_parts(input, input_len as usize)};
let mut state = cast_pointer!(state, SMBState);
let flow = cast_pointer!(flow, Flow);
SCLogDebug!("parsing {} bytes of request data", input_len);
/* START with MISTREAM set: record might be starting the middle. */
if flags & (STREAM_START|STREAM_MIDSTREAM) == (STREAM_START|STREAM_MIDSTREAM) {
state.ts_gap = true;
}
state.update_ts(flow.get_last_time().as_secs());
state.parse_tcp_data_ts(buf)
}
#[no_mangle]
pub extern "C" fn rs_smb_parse_request_tcp_gap(
state: &mut SMBState,
input_len: u32)
-> AppLayerResult
{
state.parse_tcp_data_ts_gap(input_len as u32)
}
#[no_mangle]
pub extern "C" fn rs_smb_parse_response_tcp(flow: *const Flow,
state: *mut ffi::c_void,
_pstate: *mut std::os::raw::c_void,
input: *const u8,
input_len: u32,
_data: *const ffi::c_void,
flags: u8)
-> AppLayerResult
{
let mut state = cast_pointer!(state, SMBState);
let flow = cast_pointer!(flow, Flow);
SCLogDebug!("parsing {} bytes of response data", input_len);
let buf = unsafe{std::slice::from_raw_parts(input, input_len as usize)};
/* START with MISTREAM set: record might be starting the middle. */
if flags & (STREAM_START|STREAM_MIDSTREAM) == (STREAM_START|STREAM_MIDSTREAM) {
state.tc_gap = true;
}
state.update_ts(flow.get_last_time().as_secs());
state.parse_tcp_data_tc(buf)
}
#[no_mangle]
pub extern "C" fn rs_smb_parse_response_tcp_gap(
state: &mut SMBState,
input_len: u32)
-> AppLayerResult
{
state.parse_tcp_data_tc_gap(input_len as u32)
}
fn smb_probe_tcp_midstream(direction: u8, slice: &[u8], rdir: *mut u8) -> i8
{
match search_smb_record(slice) {
Ok((_, ref data)) => {
SCLogDebug!("smb found");
match parse_smb_version(data) {
Ok((_, ref smb)) => {
SCLogDebug!("SMB {:?}", smb);
if smb.version == 0xff_u8 { // SMB1
SCLogDebug!("SMBv1 record");
match parse_smb_record(data) {
Ok((_, ref smb_record)) => {
if smb_record.flags & 0x80 != 0 {
SCLogDebug!("RESPONSE {:02x}", smb_record.flags);
if direction & STREAM_TOSERVER != 0 {
unsafe { *rdir = STREAM_TOCLIENT; }
}
} else {
SCLogDebug!("REQUEST {:02x}", smb_record.flags);
if direction & STREAM_TOCLIENT != 0 {
unsafe { *rdir = STREAM_TOSERVER; }
}
}
return 1;
},
_ => { },
}
} else if smb.version == 0xfe_u8 { // SMB2
SCLogDebug!("SMB2 record");
match parse_smb2_record_direction(data) {
Ok((_, ref smb_record)) => {
if direction & STREAM_TOSERVER != 0 {
SCLogDebug!("direction STREAM_TOSERVER smb_record {:?}", smb_record);
if !smb_record.request {
unsafe { *rdir = STREAM_TOCLIENT; }
}
} else {
SCLogDebug!("direction STREAM_TOCLIENT smb_record {:?}", smb_record);
if smb_record.request {
unsafe { *rdir = STREAM_TOSERVER; }
}
}
},
_ => {},
}
}
else if smb.version == 0xfd_u8 { // SMB3 transform
SCLogDebug!("SMB3 record");
}
return 1;
},
_ => {
SCLogDebug!("smb not found in {:?}", slice);
},
}
},
_ => {
SCLogDebug!("no dice");
},
}
return 0;
}
// probing parser
// return 1 if found, 0 is not found
#[no_mangle]
pub extern "C" fn rs_smb_probe_tcp(_f: *const Flow,
flags: u8, input: *const u8, len: u32, rdir: *mut u8)
-> AppProto
{
let slice = build_slice!(input, len as usize);
if flags & STREAM_MIDSTREAM == STREAM_MIDSTREAM {
if smb_probe_tcp_midstream(flags, slice, rdir) == 1 {
return unsafe { ALPROTO_SMB };
}
}
match parse_nbss_record_partial(slice) {
Ok((_, ref hdr)) => {
if hdr.is_smb() {
SCLogDebug!("smb found");
return unsafe { ALPROTO_SMB };
} else if hdr.needs_more(){
return 0;
} else if hdr.is_valid() &&
hdr.message_type != NBSS_MSGTYPE_SESSION_MESSAGE {
//we accept a first small netbios message before real SMB
let hl = hdr.length as usize;
if hdr.data.len() >= hl + 8 {
// 8 is 4 bytes NBSS + 4 bytes SMB0xFX magic
match parse_nbss_record_partial(&hdr.data[hl..]) {
Ok((_, ref hdr2)) => {
if hdr2.is_smb() {
SCLogDebug!("smb found");
return unsafe { ALPROTO_SMB };
}
}
_ => {}
}
} else if hdr.length < 256 {
// we want more data, 256 is some random value
return 0;
}
// default is failure
}
},
_ => { },
}
SCLogDebug!("no smb");
return unsafe { ALPROTO_FAILED };
}
#[no_mangle]
pub extern "C" fn rs_smb_state_get_tx_count(state: *mut ffi::c_void)
-> u64
{
let state = cast_pointer!(state, SMBState);
SCLogDebug!("rs_smb_state_get_tx_count: returning {}", state.tx_id);
return state.tx_id;
}
#[no_mangle]
pub extern "C" fn rs_smb_state_get_tx(state: *mut ffi::c_void,
tx_id: u64)
-> *mut ffi::c_void
{
let state = cast_pointer!(state, SMBState);
match state.get_tx_by_id(tx_id) {
Some(tx) => {
return unsafe{transmute(tx)};
}
None => {
return std::ptr::null_mut();
}
}
}
// for use with the C API call StateGetTxIterator
#[no_mangle]
pub extern "C" fn rs_smb_state_get_tx_iterator(
_ipproto: u8,
_alproto: AppProto,
state: *mut std::os::raw::c_void,
min_tx_id: u64,
_max_tx_id: u64,
istate: &mut u64,
) -> applayer::AppLayerGetTxIterTuple
{
let state = cast_pointer!(state, SMBState);
match state.get_tx_iterator(min_tx_id, istate) {
Some((tx, out_tx_id, has_next)) => {
let c_tx = unsafe { transmute(tx) };
let ires = applayer::AppLayerGetTxIterTuple::with_values(c_tx, out_tx_id, has_next);
return ires;
}
None => {
return applayer::AppLayerGetTxIterTuple::not_found();
}
}
}
#[no_mangle]
pub extern "C" fn rs_smb_state_tx_free(state: *mut ffi::c_void,
tx_id: u64)
{
let state = cast_pointer!(state, SMBState);
SCLogDebug!("freeing tx {}", tx_id as u64);
state.free_tx(tx_id);
}
#[no_mangle]
pub extern "C" fn rs_smb_tx_get_alstate_progress(tx: *mut ffi::c_void,
direction: u8)
-> i32
{
let tx = cast_pointer!(tx, SMBTransaction);
if direction == STREAM_TOSERVER && tx.request_done {
SCLogDebug!("tx {} TOSERVER progress 1 => {:?}", tx.id, tx);
return 1;
} else if direction == STREAM_TOCLIENT && tx.response_done {
SCLogDebug!("tx {} TOCLIENT progress 1 => {:?}", tx.id, tx);
return 1;
} else {
SCLogDebug!("tx {} direction {} progress 0", tx.id, direction);
return 0;
}
}
#[no_mangle]
pub extern "C" fn rs_smb_get_tx_data(
tx: *mut std::os::raw::c_void)
-> *mut AppLayerTxData
{
let tx = cast_pointer!(tx, SMBTransaction);
return &mut tx.tx_data;
}
#[no_mangle]
pub extern "C" fn rs_smb_state_get_tx_detect_state(
tx: *mut std::os::raw::c_void)
-> *mut DetectEngineState
{
let tx = cast_pointer!(tx, SMBTransaction);
match tx.de_state {
Some(ds) => {
return ds;
},
None => {
return std::ptr::null_mut();
}
}
}
#[no_mangle]
pub extern "C" fn rs_smb_state_set_tx_detect_state(
tx: *mut std::os::raw::c_void,
de_state: &mut DetectEngineState) -> std::os::raw::c_int
{
let tx = cast_pointer!(tx, SMBTransaction);
tx.de_state = Some(de_state);
0
}
#[no_mangle]
pub extern "C" fn rs_smb_state_truncate(
state: *mut std::ffi::c_void,
direction: u8)
{
let state = cast_pointer!(state, SMBState);
if (direction & STREAM_TOSERVER) != 0 {
state.trunc_ts();
} else {
state.trunc_tc();
}
}
#[no_mangle]
pub extern "C" fn rs_smb_state_get_events(tx: *mut std::os::raw::c_void)
-> *mut AppLayerDecoderEvents
{
let tx = cast_pointer!(tx, SMBTransaction);
return tx.events;
}
#[no_mangle]
pub extern "C" fn rs_smb_state_get_event_info_by_id(event_id: std::os::raw::c_int,
event_name: *mut *const std::os::raw::c_char,
event_type: *mut AppLayerEventType)
-> i8
{
if let Some(e) = SMBEvent::from_i32(event_id as i32) {
let estr = match e {
SMBEvent::InternalError => { "internal_error\0" },
SMBEvent::MalformedData => { "malformed_data\0" },
SMBEvent::RecordOverflow => { "record_overflow\0" },
SMBEvent::MalformedNtlmsspRequest => { "malformed_ntlmssp_request\0" },
SMBEvent::MalformedNtlmsspResponse => { "malformed_ntlmssp_response\0" },
SMBEvent::DuplicateNegotiate => { "duplicate_negotiate\0" },
SMBEvent::NegotiateMalformedDialects => { "netogiate_malformed_dialects\0" },
SMBEvent::FileOverlap => { "file_overlap\0" },
};
unsafe{
*event_name = estr.as_ptr() as *const std::os::raw::c_char;
*event_type = APP_LAYER_EVENT_TYPE_TRANSACTION;
};
0
} else {
-1
}
}
#[no_mangle]
pub extern "C" fn rs_smb_state_get_event_info(event_name: *const std::os::raw::c_char,
event_id: *mut std::os::raw::c_int,
event_type: *mut AppLayerEventType)
-> i32
{
if event_name == std::ptr::null() {
return -1;
}
let c_event_name: &CStr = unsafe { CStr::from_ptr(event_name) };
let event = match c_event_name.to_str() {
Ok(s) => {
smb_str_to_event(s)
},
Err(_) => -1, // UTF-8 conversion failed
};
unsafe {
*event_type = APP_LAYER_EVENT_TYPE_TRANSACTION;
*event_id = event as std::os::raw::c_int;
};
if event == -1 {
return -1;
}
0
}