libwifi.py 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. # Copyright (c) 2017, Mathy Vanhoef <Mathy.Vanhoef@cs.kuleuven.be>
  2. #
  3. # This code may be distributed under the terms of the BSD license.
  4. # See README for more details.
  5. from scapy.all import *
  6. #### Packet Processing Functions ####
  7. class MitmSocket(L2Socket):
  8. def __init__(self, **kwargs):
  9. super(MitmSocket, self).__init__(**kwargs)
  10. def send(self, p):
  11. # Hack: set the More Data flag so we can detect injected frames (and so clients stay awake longer)
  12. p[Dot11].FCfield |= 0x20
  13. L2Socket.send(self, RadioTap()/p)
  14. def _strip_fcs(self, p):
  15. # Scapy can't handle the optional Frame Check Sequence (FCS) field automatically
  16. if p[RadioTap].present & 2 != 0:
  17. rawframe = str(p[RadioTap])
  18. pos = 8
  19. while ord(rawframe[pos - 1]) & 0x80 != 0: pos += 4
  20. # If the TSFT field is present, it must be 8-bytes aligned
  21. if p[RadioTap].present & 1 != 0:
  22. pos += (8 - (pos % 8))
  23. pos += 8
  24. # Remove FCS if present
  25. if ord(rawframe[pos]) & 0x10 != 0:
  26. return Dot11(str(p[Dot11])[:-4])
  27. return p[Dot11]
  28. def recv(self, x=MTU):
  29. p = L2Socket.recv(self, x)
  30. if p == None or not Dot11 in p: return None
  31. # Hack: ignore frames that we just injected and are echoed back by the kernel
  32. if p[Dot11].FCfield & 0x20 != 0:
  33. return None
  34. # Strip the FCS if present, and drop the RadioTap header
  35. return self._strip_fcs(p)
  36. def close(self):
  37. super(MitmSocket, self).close()
  38. def dot11_get_seqnum(p):
  39. return p[Dot11].SC >> 4
  40. def dot11_get_iv(p):
  41. """Scapy can't handle Extended IVs, so do this properly ourselves (only works for CCMP)"""
  42. if Dot11WEP not in p:
  43. log(ERROR, "INTERNAL ERROR: Requested IV of plaintext frame")
  44. return 0
  45. wep = p[Dot11WEP]
  46. if wep.keyid & 32:
  47. return ord(wep.iv[0]) + (ord(wep.iv[1]) << 8) + (struct.unpack(">I", wep.wepdata[:4])[0] << 16)
  48. else:
  49. return ord(wep.iv[0]) + (ord(wep.iv[1]) << 8) + (ord(wep.iv[2]) << 16)
  50. def get_tlv_value(p, type):
  51. if not Dot11Elt in p: return None
  52. el = p[Dot11Elt]
  53. while isinstance(el, Dot11Elt):
  54. if el.ID == type:
  55. return el.info
  56. el = el.payload
  57. return None
  58. class IvInfo():
  59. def __init__(self, p):
  60. self.iv = dot11_get_iv(p)
  61. self.seq = dot11_get_seqnum(p)
  62. self.time = p.time
  63. def is_reused(self, p):
  64. """Return true if frame p reuses an IV and if p is not a retransmitted frame"""
  65. iv = dot11_get_iv(p)
  66. seq = dot11_get_seqnum(p)
  67. return self.iv == iv and self.seq != seq and p.time >= self.time + 1
  68. class IvCollection():
  69. def __init__(self):
  70. self.ivs = dict() # maps IV values to IvInfo objects
  71. def reset(self):
  72. self.ivs = dict()
  73. def track_used_iv(self, p):
  74. iv = dot11_get_iv(p)
  75. self.ivs[iv] = IvInfo(p)
  76. def is_iv_reused(self, p):
  77. """Returns True if this is an *observed* IV reuse and not just a retransmission"""
  78. iv = dot11_get_iv(p)
  79. return iv in self.ivs and self.ivs[iv].is_reused(p)
  80. def is_new_iv(self, p):
  81. """Returns True if the IV in this frame is higher than all previously observed ones"""
  82. iv = dot11_get_iv(p)
  83. if len(self.ivs) == 0: return True
  84. return iv > max(self.ivs.keys())