test_rrm.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. # Radio measurement
  2. # Copyright(c) 2013 - 2016 Intel Mobile Communications GmbH.
  3. # Copyright(c) 2011 - 2016 Intel Corporation. All rights reserved.
  4. #
  5. # This software may be distributed under the terms of the BSD license.
  6. # See README for more details.
  7. import re
  8. import logging
  9. logger = logging.getLogger()
  10. import hostapd
  11. from utils import HwsimSkip
  12. nr="00112233445500000000510107"
  13. lci="01000800101298c0b512926666f6c2f1001c00004104050000c00012"
  14. civic="01000b0011223344556677889900998877665544332211aabbccddeeff"
  15. def check_nr_results(dev, bssids=None, lci=False, civic=False):
  16. if bssids is None:
  17. ev = dev.wait_event(["RRM-NEIGHBOR-REP-REQUEST-FAILED" ], timeout=10)
  18. if ev is None:
  19. raise Exception("RRM neighbor report failure not received")
  20. return
  21. received = []
  22. for bssid in bssids:
  23. ev = dev.wait_event(["RRM-NEIGHBOR-REP-RECEIVED"], timeout=10)
  24. if ev is None:
  25. raise Exception("RRM report result not indicated")
  26. received.append(ev)
  27. for bssid in bssids:
  28. found = False
  29. for r in received:
  30. if "RRM-NEIGHBOR-REP-RECEIVED bssid=" + bssid in r:
  31. if lci and "lci=" not in r:
  32. raise Exception("LCI data not reported for %s" % bssid)
  33. if civic and "civic=" not in r:
  34. raise Exception("civic data not reported for %s" % bssid)
  35. received.remove(r)
  36. found = True
  37. break
  38. if not found:
  39. raise Exception("RRM report result for %s not indicated" % bssid)
  40. def test_rrm_neighbor_db(dev, apdev):
  41. """hostapd ctrl_iface SET_NEIGHBOR"""
  42. params = { "ssid": "test", "rrm_neighbor_report": "1" }
  43. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  44. # Bad BSSID
  45. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:gg ssid=\"test1\" nr=" + nr):
  46. raise Exception("Set neighbor succeeded unexpectedly")
  47. # Bad SSID
  48. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=test1 nr=" + nr):
  49. raise Exception("Set neighbor succeeded unexpectedly")
  50. # No SSID
  51. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 nr=" + nr):
  52. raise Exception("Set neighbor succeeded unexpectedly")
  53. # No NR
  54. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\""):
  55. raise Exception("Set neighbor succeeded unexpectedly")
  56. # Odd length of NR
  57. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr[:-1]):
  58. raise Exception("Set neighbor succeeded unexpectedly")
  59. # Invalid lci
  60. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr + " lci=1"):
  61. raise Exception("Set neighbor succeeded unexpectedly")
  62. # Invalid civic
  63. if "FAIL" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr + " civic=1"):
  64. raise Exception("Set neighbor succeeded unexpectedly")
  65. # No entry yet in database
  66. if "FAIL" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\""):
  67. raise Exception("Remove neighbor succeeded unexpectedly")
  68. # Add a neighbor entry
  69. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr + " lci=" + lci + " civic=" + civic):
  70. raise Exception("Set neighbor failed")
  71. # Another BSSID with the same SSID
  72. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:56 ssid=\"test1\" nr=" + nr + " lci=" + lci + " civic=" + civic):
  73. raise Exception("Set neighbor failed")
  74. # Fewer parameters
  75. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr):
  76. raise Exception("Set neighbor failed")
  77. # SSID in hex format
  78. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=7465737431 nr=" + nr):
  79. raise Exception("Set neighbor failed")
  80. # With more parameters
  81. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr + " civic=" + civic):
  82. raise Exception("Set neighbor failed")
  83. # With all parameters
  84. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\" nr=" + nr + " lci=" + lci + " civic=" + civic):
  85. raise Exception("Set neighbor failed")
  86. # Another SSID on the same BSSID
  87. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test2\" nr=" + nr + " lci=" + lci):
  88. raise Exception("Set neighbor failed")
  89. if "OK" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\""):
  90. raise Exception("Remove neighbor failed")
  91. if "OK" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:56 ssid=\"test1\""):
  92. raise Exception("Remove neighbor failed")
  93. if "OK" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55 ssid=\"test2\""):
  94. raise Exception("Remove neighbor failed")
  95. # Double remove
  96. if "FAIL" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1\""):
  97. raise Exception("Remove neighbor succeeded unexpectedly")
  98. # Stationary AP
  99. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test3\" nr=" + nr + " lci=" + lci + " civic=" + civic + " stat"):
  100. raise Exception("Set neighbor failed")
  101. if "OK" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55 ssid=\"test3\""):
  102. raise Exception("Remove neighbor failed")
  103. # Invalid remove - bad BSSID
  104. if "FAIL" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:5 ssid=\"test1\""):
  105. raise Exception("Remove neighbor succeeded unexpectedly")
  106. # Invalid remove - bad SSID
  107. if "FAIL" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55 ssid=\"test1"):
  108. raise Exception("Remove neighbor succeeded unexpectedly")
  109. # Invalid remove - missing SSID
  110. if "FAIL" not in hapd.request("REMOVE_NEIGHBOR 00:11:22:33:44:55"):
  111. raise Exception("Remove neighbor succeeded unexpectedly")
  112. def test_rrm_neighbor_rep_req(dev, apdev):
  113. """wpa_supplicant ctrl_iface NEIGHBOR_REP_REQUEST"""
  114. nr1="00112233445500000000510107"
  115. nr2="00112233445600000000510107"
  116. nr3="dd112233445500000000510107"
  117. params = { "ssid": "test" }
  118. hostapd.add_ap(apdev[0]['ifname'], params)
  119. params = { "ssid": "test2", "rrm_neighbor_report": "1" }
  120. hapd = hostapd.add_ap(apdev[1]['ifname'], params)
  121. bssid1 = apdev[1]['bssid']
  122. dev[0].connect("test", key_mgmt="NONE", scan_freq="2412")
  123. if "FAIL" not in dev[0].request("NEIGHBOR_REP_REQUEST"):
  124. raise Exception("Request succeeded unexpectedly (AP without RRM)")
  125. if "FAIL" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"abcdef\""):
  126. raise Exception("Request succeeded unexpectedly (AP without RRM 2)")
  127. dev[0].request("DISCONNECT")
  128. rrm = int(dev[0].get_driver_status_field("capa.rrm_flags"), 16)
  129. if rrm & 0x5 != 0x5 and rrm & 0x10 != 0x10:
  130. raise HwsimSkip("Required RRM capabilities are not supported")
  131. dev[0].connect("test2", key_mgmt="NONE", scan_freq="2412")
  132. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST"):
  133. raise Exception("Request failed")
  134. check_nr_results(dev[0], [bssid1])
  135. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST lci"):
  136. raise Exception("Request failed")
  137. check_nr_results(dev[0], [bssid1])
  138. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST lci civic"):
  139. raise Exception("Request failed")
  140. check_nr_results(dev[0], [bssid1])
  141. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test3\""):
  142. raise Exception("Request failed")
  143. check_nr_results(dev[0])
  144. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test3\" lci civic"):
  145. raise Exception("Request failed")
  146. check_nr_results(dev[0])
  147. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:55 ssid=\"test3\" nr=" + nr1 + " lci=" + lci + " civic=" + civic):
  148. raise Exception("Set neighbor failed")
  149. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:56 ssid=\"test3\" nr=" + nr2 + " lci=" + lci + " civic=" + civic):
  150. raise Exception("Set neighbor failed")
  151. if "OK" not in hapd.request("SET_NEIGHBOR 00:11:22:33:44:56 ssid=\"test4\" nr=" + nr2 + " lci=" + lci + " civic=" + civic):
  152. raise Exception("Set neighbor failed")
  153. if "OK" not in hapd.request("SET_NEIGHBOR dd:11:22:33:44:55 ssid=\"test5\" nr=" + nr3 + " lci=" + lci):
  154. raise Exception("Set neighbor failed")
  155. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test3\""):
  156. raise Exception("Request failed")
  157. check_nr_results(dev[0], ["00:11:22:33:44:55", "00:11:22:33:44:56"])
  158. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test3\" lci"):
  159. raise Exception("Request failed")
  160. check_nr_results(dev[0], ["00:11:22:33:44:55", "00:11:22:33:44:56"],
  161. lci=True)
  162. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test3\" civic"):
  163. raise Exception("Request failed")
  164. check_nr_results(dev[0], ["00:11:22:33:44:55", "00:11:22:33:44:56"],
  165. civic=True)
  166. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test3\" lci civic"):
  167. raise Exception("Request failed")
  168. check_nr_results(dev[0], ["00:11:22:33:44:55", "00:11:22:33:44:56"],
  169. lci=True, civic=True)
  170. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test4\""):
  171. raise Exception("Request failed")
  172. check_nr_results(dev[0], ["00:11:22:33:44:56"])
  173. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test4\" lci"):
  174. raise Exception("Request failed")
  175. check_nr_results(dev[0], ["00:11:22:33:44:56"], lci=True)
  176. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test4\" civic"):
  177. raise Exception("Request failed")
  178. check_nr_results(dev[0], ["00:11:22:33:44:56"], civic=True)
  179. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test4\" lci civic"):
  180. raise Exception("Request failed")
  181. check_nr_results(dev[0], ["00:11:22:33:44:56"], lci=True, civic=True)
  182. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test5\""):
  183. raise Exception("Request failed")
  184. check_nr_results(dev[0], ["dd:11:22:33:44:55"])
  185. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test5\" lci"):
  186. raise Exception("Request failed")
  187. check_nr_results(dev[0], ["dd:11:22:33:44:55"], lci=True)
  188. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test5\" civic"):
  189. raise Exception("Request failed")
  190. check_nr_results(dev[0], ["dd:11:22:33:44:55"])
  191. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST ssid=\"test5\" lci civic"):
  192. raise Exception("Request failed")
  193. check_nr_results(dev[0], ["dd:11:22:33:44:55"], lci=True)
  194. def test_rrm_lci_req(dev, apdev):
  195. """hostapd lci request"""
  196. rrm = int(dev[0].get_driver_status_field("capa.rrm_flags"), 16)
  197. if rrm & 0x5 != 0x5 and rrm & 0x10 != 0x10:
  198. raise HwsimSkip("Required RRM capabilities are not supported")
  199. params = { "ssid": "rrm", "rrm_neighbor_report": "1" }
  200. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  201. # station not specified
  202. if "FAIL" not in hapd.request("REQ_LCI "):
  203. raise Exception("REQ_LCI with no station succeeded unexpectedly")
  204. # station that is not connected specified
  205. if "FAIL" not in hapd.request("REQ_LCI " + dev[0].own_addr()):
  206. raise Exception("REQ_LCI succeeded unexpectedly (station not connected)")
  207. dev[0].request("SET LCI ")
  208. dev[0].connect("rrm", key_mgmt="NONE", scan_freq="2412")
  209. # station connected without LCI
  210. if "FAIL" not in hapd.request("REQ_LCI " + dev[0].own_addr()):
  211. raise Exception("REQ_LCI succeeded unexpectedly (station without lci)")
  212. dev[0].request("DISCONNECT")
  213. dev[0].wait_disconnected(timeout=2)
  214. dev[0].request("SET LCI " + lci)
  215. dev[0].connect("rrm", key_mgmt="NONE", scan_freq="2412")
  216. # station connected with LCI
  217. if "OK" not in hapd.request("REQ_LCI " + dev[0].own_addr()):
  218. raise Exception("REQ_LCI failed unexpectedly")
  219. def test_rrm_neighbor_rep_req_from_conf(dev, apdev):
  220. """wpa_supplicant ctrl_iface NEIGHBOR_REP_REQUEST and hostapd config"""
  221. params = { "ssid": "test2", "rrm_neighbor_report": "1",
  222. "stationary_ap": "1", "lci": lci, "civic": civic }
  223. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  224. bssid = apdev[0]['bssid']
  225. rrm = int(dev[0].get_driver_status_field("capa.rrm_flags"), 16)
  226. if rrm & 0x5 != 0x5 and rrm & 0x10 != 0x10:
  227. raise HwsimSkip("Required RRM capabilities are not supported")
  228. dev[0].connect("test2", key_mgmt="NONE", scan_freq="2412")
  229. if "OK" not in dev[0].request("NEIGHBOR_REP_REQUEST"):
  230. raise Exception("Request failed")
  231. check_nr_results(dev[0], [bssid])
  232. def test_rrm_ftm_range_req(dev, apdev):
  233. """hostapd FTM range request command"""
  234. rrm = int(dev[0].get_driver_status_field("capa.rrm_flags"), 16)
  235. if rrm & 0x5 != 0x5 and rrm & 0x10 != 0x10:
  236. raise HwsimSkip("Required RRM capabilities are not supported")
  237. params = { "ssid": "rrm", "rrm_neighbor_report": "1" }
  238. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  239. # station not specified
  240. if "FAIL" not in hapd.request("REQ_RANGE "):
  241. raise Exception("REQ_RANGE with no station succeeded unexpectedly")
  242. # station that is not connected specified
  243. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr()):
  244. raise Exception("REQ_RANGE succeeded unexpectedly (station not connected)")
  245. # No responders specified
  246. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10 10"):
  247. raise Exception("REQ_RANGE succeeded unexpectedly (no responder)")
  248. # Bad responder address
  249. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10 10 00:11:22:33:44:"):
  250. raise Exception("REQ_RANGE succeeded unexpectedly (bad responder address)")
  251. # Bad responder address
  252. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10 10 00:11:22:33:44:55 00:11:22:33:44"):
  253. raise Exception("REQ_RANGE succeeded unexpectedly (bad responder address 2)")
  254. # Bad min_ap value
  255. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10 300 00:11:22:33:44:55"):
  256. raise Exception("REQ_RANGE succeeded unexpectedly (invalid min_ap value)")
  257. # Bad rand value
  258. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " -1 10 00:11:22:33:44:55"):
  259. raise Exception("REQ_RANGE succeeded unexpectedly (invalid rand value)")
  260. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 65536 10 00:11:22:33:44:55"):
  261. raise Exception("REQ_RANGE succeeded unexpectedly (invalid rand value)")
  262. # Missing min_ap value
  263. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10"):
  264. raise Exception("REQ_RANGE succeeded unexpectedly (missing min_ap value)")
  265. # Too many responders
  266. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10 10" + 20*" 00:11:22:33:44:55"):
  267. raise Exception("REQ_RANGE succeeded unexpectedly (too many responders)")
  268. dev[0].connect("rrm", key_mgmt="NONE", scan_freq="2412")
  269. # Responder not in database
  270. # Note: this check would pass since the station does not support FTM range
  271. # request and not because the responder is not in the database.
  272. if "FAIL" not in hapd.request("REQ_RANGE " + dev[0].own_addr() + " 10 10 00:11:22:33:44:55"):
  273. raise Exception("REQ_RANGE succeeded unexpectedly (responder not in database)")
  274. def test_rrm_ftm_capa_indication(dev, apdev):
  275. """FTM capability indication"""
  276. try:
  277. _test_rrm_ftm_capa_indication(dev, apdev)
  278. finally:
  279. dev[0].request("SET ftm_initiator 0")
  280. dev[0].request("SET ftm_responder 0")
  281. def _test_rrm_ftm_capa_indication(dev, apdev):
  282. params = { "ssid": "ftm",
  283. "ftm_responder": "1",
  284. "ftm_initiator": "1", }
  285. hapd = hostapd.add_ap(apdev[0]['ifname'], params)
  286. if "OK" not in dev[0].request("SET ftm_initiator 1"):
  287. raise Exception("could not set ftm_initiator")
  288. if "OK" not in dev[0].request("SET ftm_responder 1"):
  289. raise Exception("could not set ftm_responder")
  290. dev[0].scan_for_bss(apdev[0]['bssid'], freq=2412, force_scan=True)