112 lines
4.1 KiB
Ruby
112 lines
4.1 KiB
Ruby
# -*- coding: binary -*-
|
|
# Copyright (c) 2010, patrickHVE@googlemail.com
|
|
# All rights reserved.
|
|
#
|
|
# Redistribution and use in source and binary forms, with or without
|
|
# modification, are permitted provided that the following conditions are met:
|
|
# * Redistributions of source code must retain the above copyright
|
|
# notice, this list of conditions and the following disclaimer.
|
|
# * Redistributions in binary form must reproduce the above copyright
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
# documentation and/or other materials provided with the distribution.
|
|
# * The names of the author may not be used to endorse or promote products
|
|
# derived from this software without specific prior written permission.
|
|
#
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
# DISCLAIMED. IN NO EVENT SHALL patrickHVE@googlemail.com BE LIABLE FOR ANY
|
|
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
module Rex
|
|
module Post
|
|
module Meterpreter
|
|
module Extensions
|
|
module Stdapi
|
|
module Railgun
|
|
|
|
#
|
|
# represents one function, e.g. MessageBoxW
|
|
#
|
|
class LibraryFunction
|
|
@@allowed_datatypes = {
|
|
'VOID' => ['return'],
|
|
'BOOL' => ['in', 'return'],
|
|
'BYTE' => ['in', 'return'],
|
|
'WORD' => ['in', 'return'],
|
|
'DWORD' => ['in', 'return'],
|
|
'LPVOID' => ['in', 'return'], # sf: for specifying a memory address (e.g. VirtualAlloc/HeapAlloc/...) where we don't want to back it up with actual mem ala PBLOB
|
|
'ULONG_PTR' => ['in', 'return'],
|
|
'PDWORD' => ['in', 'out', 'inout'], # todo: support for functions that return pointers to strings
|
|
'PULONG_PTR' => ['in', 'out', 'inout', 'return'],
|
|
'PWCHAR' => ['in', 'out', 'inout', 'return'],
|
|
'PCHAR' => ['in', 'out', 'inout', 'return'],
|
|
'PBLOB' => ['in', 'out', 'inout'],
|
|
}.freeze
|
|
|
|
@@allowed_convs = ['stdcall', 'cdecl']
|
|
|
|
@@directions = ['in', 'out', 'inout', 'return'].freeze
|
|
|
|
attr_reader :return_type, :params, :remote_name, :calling_conv
|
|
|
|
def initialize(return_type, params, remote_name, calling_conv='stdcall')
|
|
check_return_type(return_type) # we do error checking as early as possible so the library is easier to use
|
|
check_params(params)
|
|
check_calling_conv(calling_conv)
|
|
@return_type = return_type
|
|
@params = params
|
|
@remote_name = remote_name
|
|
@calling_conv = calling_conv
|
|
end
|
|
|
|
private
|
|
|
|
def check_calling_conv(conv)
|
|
if not @@allowed_convs.include?(conv)
|
|
raise ArgumentError, "Calling convention unknown: #{conv}."
|
|
end
|
|
end
|
|
|
|
def check_type_exists(type)
|
|
if not @@allowed_datatypes.has_key?(type)
|
|
raise ArgumentError, "Type unknown: #{type}. Allowed types: #{PP.pp(@@allowed_datatypes.keys, "")}"
|
|
end
|
|
end
|
|
|
|
def check_return_type(type)
|
|
check_type_exists(type)
|
|
if not @@allowed_datatypes[type].include?("return")
|
|
raise ArgumentError, "#{type} is not allowed as a return type"
|
|
end
|
|
end
|
|
|
|
def check_params(params)
|
|
params.each do |param|
|
|
raise ArgumentError, "each param must be described by a three-tuple [type,name,direction]" unless param.length == 3
|
|
type = param[0]
|
|
direction = param[2]
|
|
|
|
# Assert a valid type
|
|
check_type_exists(type)
|
|
|
|
# Only our set of predefined directions are valid
|
|
unless @@directions.include?(direction)
|
|
raise ArgumentError, "invalid direction: #{direction}"
|
|
end
|
|
|
|
# 'return' is not a valid direction in this context
|
|
if direction == "return"
|
|
raise "direction 'return' is only for the return value of the function."
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
end; end; end; end; end; end
|