Class: Rex::Encoding::Xor::Generic

Inherits:
Object
  • Object
show all
Defined in:
lib/rex/encoding/xor/generic.rb

Direct Known Subclasses

Byte, Dword, DwordAdditive, Qword, Word

Class Method Summary collapse

Class Method Details

._check(data, key, badchars) ⇒ Object

hook stylies! return index of offending byte or nil



25
26
27
# File 'lib/rex/encoding/xor/generic.rb', line 25

def Generic._check(data, key, badchars)
  return _check_key(key, badchars) || _check_encode(data, key, badchars)
end

._check_encode(data, key, badchars) ⇒ Object



31
32
33
# File 'lib/rex/encoding/xor/generic.rb', line 31

def Generic._check_encode(data, key, badchars)
  return Rex::Text.badchar_index(encode(data, key), badchars)
end

._check_key(key, badchars) ⇒ Object



28
29
30
# File 'lib/rex/encoding/xor/generic.rb', line 28

def Generic._check_key(key, badchars)
  return Rex::Text.badchar_index(key, badchars)
end

._encode_mutate_key(buf, key, pos, len) ⇒ Object

kind of ghetto, but very convenient for mutating keys by default, do no key mutations



134
135
136
# File 'lib/rex/encoding/xor/generic.rb', line 134

def Generic._encode_mutate_key(buf, key, pos, len)
  return key
end

._find_bad_keys(data, badchars) ⇒ Object

Find a list of bytes that can’t be valid xor keys, from the data and badchars. This returns a Array of hashes, length keysize



45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
# File 'lib/rex/encoding/xor/generic.rb', line 45

def Generic._find_bad_keys(data, badchars)

  ksize = keysize

  # array of hashes for the bad characters based
  # on their position in the data
  badkeys = [ ]
  ksize.times { badkeys << { } }

  badchars.each_byte { |badchar|
    pos = 0
    data.each_byte { |char|
      badkeys[pos % ksize][char ^ badchar] = true
      pos += 1
    }
  }

  return badkeys
end

._find_good_key(data, badkeys, badchars) ⇒ Object

(Hopefully) find a good key, from badkeys and badchars



68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
# File 'lib/rex/encoding/xor/generic.rb', line 68

def Generic._find_good_key(data, badkeys, badchars)

  ksize = keysize
  strip = 0
  key   = ""

  while strip < keysize

    kbyte = rand(256)

    catch(:found_kbyte) do
      256.times {

        if !badkeys[strip][kbyte] && !badchars[kbyte.chr]
          throw :found_kbyte
        end

        kbyte = (kbyte + 1) & 0xff
      }

      raise KeySearchError, "Exhausted byte space for strip #{strip}!", caller
    end

    key << kbyte
    strip += 1
  end

  # ok, we should have a good key now, lets double check...
  if _check(data, key, badchars)
    raise KeySearchError, "Key found, but bad character check failed!", caller
  end

  return key
end

.encode(buf, key) ⇒ Object



103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
# File 'lib/rex/encoding/xor/generic.rb', line 103

def Generic.encode(buf, key)

  if !key.kind_of?(String)
    raise ::ArgumentError, "Key must be a string!", caller
  end

  len = key.length

  if len == 0
    raise ::ArgumentError, "Zero key length!", caller
  end

  if keysize != 0 && keysize != len
    raise ::ArgumentError, "Key length #{len}, expected #{keysize}", caller
  end

  encoded = ""
  pos     = 0

  while pos < buf.length
    encoded += (buf[pos,1].unpack("C*")[0] ^ key[pos % len, 1].unpack("C*")[0]).chr
    key = _encode_mutate_key(buf, key, pos, len)
    pos += 1
  end

  return [ encoded, key ]

end

.find_key(data, badchars) ⇒ Object



35
36
37
# File 'lib/rex/encoding/xor/generic.rb', line 35

def Generic.find_key(data, badchars)
  return _find_good_key(data, _find_bad_keys(data, badchars), badchars)
end

.find_key_and_encode(data, badchars) ⇒ Object

maybe a bit a smaller of method name?



139
140
141
142
143
# File 'lib/rex/encoding/xor/generic.rb', line 139

def Generic.find_key_and_encode(data, badchars)
  key        = find_key(data, badchars)
  enc, fkey  = encode(data, key)
  return [ enc, key, fkey ]
end

.keysizeObject



12
13
14
15
16
17
# File 'lib/rex/encoding/xor/generic.rb', line 12

def Generic.keysize
  # special case:
  # 0 means we encode based on the length of the key
  # we don't enforce any perticular key length
  return 0
end