Files
funstraw/nl80211.py

315 lines
8.0 KiB
Python
Raw Normal View History

2015-01-06 06:00:55 +00:00
#!/usr/bin/python2.7
import os
import random
2015-01-06 06:00:55 +00:00
import socket
import struct
class Iterator(object):
def __init__(self, data, offset=0, length=None):
2015-01-06 06:00:55 +00:00
self.data = data
self.offset = offset
self.length = len(self.data) if length is None else length
assert self.length <= len(self.data)
2015-01-06 06:00:55 +00:00
def __str__(self):
data = self.data[self.offset:self.length]
2015-01-06 06:00:55 +00:00
return '(%d bytes): %r' % (len(data), data)
def Advance(self, offset_incr):
assert offset_incr <= self.Remaining(), 'Want %d bytes, have %d' % (offset_incr, self.Remaining())
2015-01-06 06:00:55 +00:00
self.offset += offset_incr
def Extract(self, length):
assert length <= self.Remaining(), 'Want %d bytes, have %d' % (length, self.Remaining())
ret = self.data[self.offset:self.offset + length]
self.Advance(length)
return ret
def ExtractIterator(self, length):
assert length <= self.Remaining(), 'Want %d bytes, have %d' % (length, self.Remaining())
ret = Iterator(self.data, self.offset, self.offset + length)
self.Advance(length)
return ret
def Remaining(self):
return self.length - self.offset
2015-01-06 06:00:55 +00:00
def AtEnd(self):
return not self.Remaining()
2015-01-06 06:00:55 +00:00
class Accumulator(object):
def __init__(self):
self._parts = []
def __str__(self):
return ''.join(self._parts)
def __len__(self):
return sum(len(part) for part in self._parts)
def Append(self, value):
self._parts.append(value)
class SingleStructParser(struct.Struct):
def Unpack(self, iterator):
values = self.unpack_from(iterator.data, iterator.offset)
iterator.Advance(self.size)
assert len(values) == 1
return values[0]
def Pack(self, accumulator, value):
accumulator.Append(self.pack(value))
class StructParser(struct.Struct):
2015-01-06 06:00:55 +00:00
def __init__(self, format, fields=None):
super(StructParser, self).__init__(format)
2015-01-06 06:00:55 +00:00
self._fields = fields
def Unpack(self, iterator):
values = self.unpack_from(iterator.data, iterator.offset)
iterator.Advance(self.size)
return dict(zip(self._fields, values))
def Pack(self, accumulator, **values):
ordered_values = []
for field in self._fields:
ordered_values.append(values[field])
accumulator.Append(self.pack(*ordered_values))
2015-01-06 06:00:55 +00:00
class StringParser(object):
def Unpack(self, iterator):
return iterator.Extract(iterator.Remaining())
def Pack(self, accumulator, value):
accumulator.Append(value)
2015-01-06 06:00:55 +00:00
class EmptyParser(object):
def Unpack(self, iterator):
2015-01-06 06:00:55 +00:00
return True
def Pack(self, accumulator, value=None):
pass
2015-01-06 06:00:55 +00:00
nlmsghdr = StructParser('LHHLL', ('length', 'type', 'flags', 'seq', 'pid'))
genlmsghdr = StructParser('BBH', ('cmd', 'version', 'reserved'))
2015-01-06 06:00:55 +00:00
class Attribute(object):
_nlattr = StructParser('HH', ('len', 'type'))
2015-01-06 06:00:55 +00:00
def __init__(self, attributes):
super(Attribute, self).__init__()
self._attributes = attributes
def Unpack(self, iterator):
nlattr = self._nlattr.Unpack(iterator)
2015-01-08 06:09:38 +00:00
value = iterator.data[iterator.offset:iterator.offset + nlattr['len'] - self._nlattr.size]
2015-01-06 06:00:55 +00:00
name, sub_parser = self._attributes.get(nlattr['type'], (None, None))
assert sub_parser, 'Unknown attribute type %d, len %d' % (nlattr['type'], nlattr['len'])
sub_len = nlattr['len'] - self._nlattr.size
sub_iterator = iterator.ExtractIterator(sub_len)
2015-01-06 06:00:55 +00:00
ret = {
name: sub_parser.Unpack(sub_iterator)
2015-01-06 06:00:55 +00:00
}
assert sub_iterator.AtEnd(), '%d bytes remaining' % sub_iterator.Remaining()
2015-01-06 06:00:55 +00:00
padding = ((nlattr['len'] + 4 - 1) & ~3) - nlattr['len']
iterator.Advance(padding)
2015-01-06 06:00:55 +00:00
return ret
def Pack(self, accumulator, attrtype, value):
sub_parser = self._attributes[attrtype][1]
sub_accumulator = Accumulator()
sub_parser.Pack(sub_accumulator, value)
self._nlattr.Pack(accumulator, len=self._nlattr.size + len(sub_accumulator), type=attrtype)
accumulator.Append(str(sub_accumulator))
2015-01-06 06:00:55 +00:00
class Attributes(object):
2015-01-06 06:00:55 +00:00
def __init__(self, attributes):
super(Attributes, self).__init__()
self._attribute_idx = dict((v[0], k) for k, v in attributes.iteritems())
2015-01-06 06:00:55 +00:00
self._attribute = Attribute(attributes)
def Unpack(self, iterator):
ret = {}
while not iterator.AtEnd():
ret.update(self._attribute.Unpack(iterator))
2015-01-06 06:00:55 +00:00
return ret
def Pack(self, accumulator, **attrs):
for name, value in attrs.iteritems():
self._attribute.Pack(accumulator, self._attribute_idx[name], value)
2015-01-06 06:00:55 +00:00
2015-01-08 06:09:38 +00:00
class Array(object):
_arrayhdr = StructParser('HH', ('len', 'index'))
def __init__(self, child):
super(Array, self).__init__()
self._child = child
def Unpack(self, iterator):
2015-01-08 06:09:38 +00:00
ret = []
while not iterator.AtEnd():
hdr = self._arrayhdr.Unpack(iterator)
sub_len = hdr['len'] - self._arrayhdr.size
sub_iterator = iterator.ExtractIterator(sub_len)
ret.append(self._child.Unpack(sub_iterator))
assert sub_iterator.AtEnd(), '%d bytes remaining' % sub_iterator.Remaining()
2015-01-08 06:09:38 +00:00
return ret
2015-01-06 06:00:55 +00:00
flag = EmptyParser()
string = StringParser()
u8 = SingleStructParser('B')
u16 = SingleStructParser('H')
u32 = SingleStructParser('L')
u64 = SingleStructParser('Q')
2015-01-06 06:00:55 +00:00
rate_info = Attributes({
1: ('bitrate', u16),
2: ('mcs', u8),
4: ('short_gi', flag),
5: ('bitrate32', u32),
9: ('80p80_mhz_width', u32),
10: ('160_mhz_width', u32),
})
bss_param = Attributes({
2: ('short_preamble', flag),
3: ('short_slot_time', flag),
4: ('dtim_period', u8),
5: ('beacon_interval', u16),
})
sta_info = Attributes({
1: ('inactive_time', u32),
2: ('rx_bytes', u32),
3: ('tx_bytes', u32),
7: ('signal', u8),
8: ('tx_bitrate', rate_info),
9: ('rx_packets', u32),
10: ('tx_packets', u32),
11: ('tx_retries', u32),
12: ('tx_failed', u32),
13: ('signal_avg', u8),
14: ('rx_bitrate', rate_info),
15: ('bss_param', bss_param),
16: ('connected_time', u32),
17: ('sta_flags', StructParser('LL', ('mask', 'values'))),
2015-01-06 06:00:55 +00:00
18: ('beacon_loss', u32),
23: ('rx_bytes_64', u64),
24: ('tx_bytes_64', u64),
})
nl80211_attr = Attributes({
3: ('ifindex', u32),
6: ('mac', string),
21: ('sta_info', sta_info),
46: ('generation', u32),
})
2015-01-08 06:09:38 +00:00
op_attr = Attributes({
1: ('id', u32),
2: ('flags', u32),
})
mcast_grp_attr = Attributes({
1: ('name', string),
2: ('id', u32),
})
ctrl_attr = Attributes({
1: ('family_id', u16),
2: ('family_name', string),
3: ('version', u32),
4: ('hdrsize', u32),
5: ('maxattr', u32),
6: ('ops', Array(op_attr)),
7: ('mcast_groups', Array(mcast_grp_attr)),
})
F_REQUEST = 1 << 0
F_MULTI = 1 << 1
F_ACK = 1 << 2
F_ECHO = 1 << 3
F_DUMP_INTR = 1 << 4
F_ROOT = 1 << 8
F_MATCH = 1 << 9
F_ATOMIC = 1 << 10
F_DUMP = F_ROOT | F_MATCH
CMD_GET_STATION = 17
STA_FLAG_AUTHORIZED = 1 << 0
STA_FLAG_SHORT_PREAMBLE = 1 << 1
STA_FLAG_WME = 1 << 2
STA_FLAG_MFP = 1 << 3
STA_FLAG_AUTHENTICATED = 1 << 4
STA_FLAG_TDLS_PEER = 1 << 5
STA_FLAG_ASSOCIATED = 1 << 6
class Connection(object):
def __init__(self):
self._sock = socket.socket(socket.AF_NETLINK, socket.SOCK_DGRAM, 16)
self._sock.bind((0, 0))
def Send(self, msg):
self._sock.send(msg)
def RecvAndUnpack(self):
data = self._sock.recv(4096)
iterator = Iterator(data)
myhdr = nlmsghdr.Unpack(iterator)
print 'nlmsghdr: %s' % myhdr
int_iterator = iterator.ExtractIterator(myhdr['length'] - nlmsghdr.size)
print 'genlmsghdr: %s' % genlmsghdr.Unpack(int_iterator)
print 'ctrl_attr: %s' % ctrl_attr.Unpack(int_iterator)
#int_genquery = Accumulator()
#genlmsghdr.Pack(
# int_genquery,
# cmd=CMD_GET_STATION,
# version=0,
# reserved=0)
#nl80211_attr.Pack(
# int_genquery,
# ifindex=6)
#genquery = Accumulator()
#nlmsghdr.Pack(
# genquery,
# length=nlmsghdr.size + len(int_genquery),
# type=20, # XXX
# flags=F_REQUEST | F_ACK | F_DUMP,
# seq=random.randint(0, 2 ** 32 - 1),
# pid=os.getpid())
#genquery.Append(str(int_genquery))
2015-01-08 06:09:38 +00:00
#sock.send(str(genquery))
#data = sock.recv(4096)
2015-01-08 06:09:38 +00:00
#iterator = Iterator(data)
#print 'nlmsghdr: %s' % nlmsghdr.Unpack(iterator)
#print 'genlmsghdr: %s' % genlmsghdr.Unpack(iterator)
#print 'nl80211_attr: %s' % nl80211_attr.Unpack(iterator)
query = '\24\0\0\0\20\0\5\3a\6\256T\v\17\0\0\3\1\0\0'
conn = Connection()
conn.Send(query)
conn.RecvAndUnpack()