1 // SPDX-License-Identifier: GPL-2.0-only 2 /* 3 * mac80211 - channel management 4 * Copyright 2020 - 2024 Intel Corporation 5 */ 6 7 #include <linux/nl80211.h> 8 #include <linux/export.h> 9 #include <linux/rtnetlink.h> 10 #include <net/cfg80211.h> 11 #include "ieee80211_i.h" 12 #include "driver-ops.h" 13 #include "rate.h" 14 15 static int ieee80211_chanctx_num_assigned(struct ieee80211_local *local, 16 struct ieee80211_chanctx *ctx) 17 { 18 struct ieee80211_link_data *link; 19 int num = 0; 20 21 lockdep_assert_wiphy(local->hw.wiphy); 22 23 list_for_each_entry(link, &ctx->assigned_links, assigned_chanctx_list) 24 num++; 25 26 return num; 27 } 28 29 static int ieee80211_chanctx_num_reserved(struct ieee80211_local *local, 30 struct ieee80211_chanctx *ctx) 31 { 32 struct ieee80211_link_data *link; 33 int num = 0; 34 35 lockdep_assert_wiphy(local->hw.wiphy); 36 37 list_for_each_entry(link, &ctx->reserved_links, reserved_chanctx_list) 38 num++; 39 40 return num; 41 } 42 43 int ieee80211_chanctx_refcount(struct ieee80211_local *local, 44 struct ieee80211_chanctx *ctx) 45 { 46 return ieee80211_chanctx_num_assigned(local, ctx) + 47 ieee80211_chanctx_num_reserved(local, ctx); 48 } 49 50 static int ieee80211_num_chanctx(struct ieee80211_local *local) 51 { 52 struct ieee80211_chanctx *ctx; 53 int num = 0; 54 55 lockdep_assert_wiphy(local->hw.wiphy); 56 57 list_for_each_entry(ctx, &local->chanctx_list, list) 58 num++; 59 60 return num; 61 } 62 63 static bool ieee80211_can_create_new_chanctx(struct ieee80211_local *local) 64 { 65 lockdep_assert_wiphy(local->hw.wiphy); 66 67 return ieee80211_num_chanctx(local) < ieee80211_max_num_channels(local); 68 } 69 70 static struct ieee80211_chanctx * 71 ieee80211_link_get_chanctx(struct ieee80211_link_data *link) 72 { 73 struct ieee80211_local *local __maybe_unused = link->sdata->local; 74 struct ieee80211_chanctx_conf *conf; 75 76 conf = rcu_dereference_protected(link->conf->chanctx_conf, 77 lockdep_is_held(&local->hw.wiphy->mtx)); 78 if (!conf) 79 return NULL; 80 81 return container_of(conf, struct ieee80211_chanctx, conf); 82 } 83 84 bool ieee80211_chanreq_identical(const struct ieee80211_chan_req *a, 85 const struct ieee80211_chan_req *b) 86 { 87 if (!cfg80211_chandef_identical(&a->oper, &b->oper)) 88 return false; 89 if (!a->ap.chan && !b->ap.chan) 90 return true; 91 return cfg80211_chandef_identical(&a->ap, &b->ap); 92 } 93 94 static const struct ieee80211_chan_req * 95 ieee80211_chanreq_compatible(const struct ieee80211_chan_req *a, 96 const struct ieee80211_chan_req *b, 97 struct ieee80211_chan_req *tmp) 98 { 99 const struct cfg80211_chan_def *compat; 100 101 if (a->ap.chan && b->ap.chan && 102 !cfg80211_chandef_identical(&a->ap, &b->ap)) 103 return NULL; 104 105 compat = cfg80211_chandef_compatible(&a->oper, &b->oper); 106 if (!compat) 107 return NULL; 108 109 /* Note: later code assumes this always fills & returns tmp if compat */ 110 tmp->oper = *compat; 111 tmp->ap = a->ap.chan ? a->ap : b->ap; 112 return tmp; 113 } 114 115 static const struct ieee80211_chan_req * 116 ieee80211_chanctx_compatible(struct ieee80211_chanctx *ctx, 117 const struct ieee80211_chan_req *req, 118 struct ieee80211_chan_req *tmp) 119 { 120 const struct ieee80211_chan_req *ret; 121 struct ieee80211_chan_req tmp2; 122 123 *tmp = (struct ieee80211_chan_req){ 124 .oper = ctx->conf.def, 125 .ap = ctx->conf.ap, 126 }; 127 128 ret = ieee80211_chanreq_compatible(tmp, req, &tmp2); 129 if (!ret) 130 return NULL; 131 *tmp = *ret; 132 return tmp; 133 } 134 135 static const struct ieee80211_chan_req * 136 ieee80211_chanctx_reserved_chanreq(struct ieee80211_local *local, 137 struct ieee80211_chanctx *ctx, 138 const struct ieee80211_chan_req *req, 139 struct ieee80211_chan_req *tmp) 140 { 141 struct ieee80211_link_data *link; 142 143 lockdep_assert_wiphy(local->hw.wiphy); 144 145 if (WARN_ON(!req)) 146 return NULL; 147 148 list_for_each_entry(link, &ctx->reserved_links, reserved_chanctx_list) { 149 req = ieee80211_chanreq_compatible(&link->reserved, req, tmp); 150 if (!req) 151 break; 152 } 153 154 return req; 155 } 156 157 static const struct ieee80211_chan_req * 158 ieee80211_chanctx_non_reserved_chandef(struct ieee80211_local *local, 159 struct ieee80211_chanctx *ctx, 160 const struct ieee80211_chan_req *compat, 161 struct ieee80211_chan_req *tmp) 162 { 163 struct ieee80211_link_data *link; 164 const struct ieee80211_chan_req *comp_def = compat; 165 166 lockdep_assert_wiphy(local->hw.wiphy); 167 168 list_for_each_entry(link, &ctx->assigned_links, assigned_chanctx_list) { 169 struct ieee80211_bss_conf *link_conf = link->conf; 170 171 if (link->reserved_chanctx) 172 continue; 173 174 comp_def = ieee80211_chanreq_compatible(&link_conf->chanreq, 175 comp_def, tmp); 176 if (!comp_def) 177 break; 178 } 179 180 return comp_def; 181 } 182 183 static bool 184 ieee80211_chanctx_can_reserve(struct ieee80211_local *local, 185 struct ieee80211_chanctx *ctx, 186 const struct ieee80211_chan_req *req) 187 { 188 struct ieee80211_chan_req tmp; 189 190 lockdep_assert_wiphy(local->hw.wiphy); 191 192 if (!ieee80211_chanctx_reserved_chanreq(local, ctx, req, &tmp)) 193 return false; 194 195 if (!ieee80211_chanctx_non_reserved_chandef(local, ctx, req, &tmp)) 196 return false; 197 198 if (!list_empty(&ctx->reserved_links) && 199 ieee80211_chanctx_reserved_chanreq(local, ctx, req, &tmp)) 200 return true; 201 202 return false; 203 } 204 205 static struct ieee80211_chanctx * 206 ieee80211_find_reservation_chanctx(struct ieee80211_local *local, 207 const struct ieee80211_chan_req *chanreq, 208 enum ieee80211_chanctx_mode mode) 209 { 210 struct ieee80211_chanctx *ctx; 211 212 lockdep_assert_wiphy(local->hw.wiphy); 213 214 if (mode == IEEE80211_CHANCTX_EXCLUSIVE) 215 return NULL; 216 217 list_for_each_entry(ctx, &local->chanctx_list, list) { 218 if (ctx->replace_state == IEEE80211_CHANCTX_WILL_BE_REPLACED) 219 continue; 220 221 if (ctx->mode == IEEE80211_CHANCTX_EXCLUSIVE) 222 continue; 223 224 if (!ieee80211_chanctx_can_reserve(local, ctx, chanreq)) 225 continue; 226 227 return ctx; 228 } 229 230 return NULL; 231 } 232 233 static enum nl80211_chan_width ieee80211_get_sta_bw(struct sta_info *sta, 234 unsigned int link_id) 235 { 236 enum ieee80211_sta_rx_bandwidth width; 237 struct link_sta_info *link_sta; 238 239 link_sta = wiphy_dereference(sta->local->hw.wiphy, sta->link[link_id]); 240 241 /* no effect if this STA has no presence on this link */ 242 if (!link_sta) 243 return NL80211_CHAN_WIDTH_20_NOHT; 244 245 width = ieee80211_sta_cap_rx_bw(link_sta); 246 247 switch (width) { 248 case IEEE80211_STA_RX_BW_20: 249 if (link_sta->pub->ht_cap.ht_supported) 250 return NL80211_CHAN_WIDTH_20; 251 else 252 return NL80211_CHAN_WIDTH_20_NOHT; 253 case IEEE80211_STA_RX_BW_40: 254 return NL80211_CHAN_WIDTH_40; 255 case IEEE80211_STA_RX_BW_80: 256 return NL80211_CHAN_WIDTH_80; 257 case IEEE80211_STA_RX_BW_160: 258 /* 259 * This applied for both 160 and 80+80. since we use 260 * the returned value to consider degradation of 261 * ctx->conf.min_def, we have to make sure to take 262 * the bigger one (NL80211_CHAN_WIDTH_160). 263 * Otherwise we might try degrading even when not 264 * needed, as the max required sta_bw returned (80+80) 265 * might be smaller than the configured bw (160). 266 */ 267 return NL80211_CHAN_WIDTH_160; 268 case IEEE80211_STA_RX_BW_320: 269 return NL80211_CHAN_WIDTH_320; 270 default: 271 WARN_ON(1); 272 return NL80211_CHAN_WIDTH_20; 273 } 274 } 275 276 static enum nl80211_chan_width 277 ieee80211_get_max_required_bw(struct ieee80211_link_data *link) 278 { 279 struct ieee80211_sub_if_data *sdata = link->sdata; 280 unsigned int link_id = link->link_id; 281 enum nl80211_chan_width max_bw = NL80211_CHAN_WIDTH_20_NOHT; 282 struct sta_info *sta; 283 284 list_for_each_entry_rcu(sta, &sdata->local->sta_list, list) { 285 if (sdata != sta->sdata && 286 !(sta->sdata->bss && sta->sdata->bss == sdata->bss)) 287 continue; 288 289 max_bw = max(max_bw, ieee80211_get_sta_bw(sta, link_id)); 290 } 291 292 return max_bw; 293 } 294 295 static enum nl80211_chan_width 296 ieee80211_get_chanctx_max_required_bw(struct ieee80211_local *local, 297 struct ieee80211_chanctx *ctx, 298 struct ieee80211_link_data *rsvd_for) 299 { 300 struct ieee80211_sub_if_data *sdata; 301 struct ieee80211_link_data *link; 302 enum nl80211_chan_width max_bw = NL80211_CHAN_WIDTH_20_NOHT; 303 304 for_each_sdata_link(local, link) { 305 enum nl80211_chan_width width = NL80211_CHAN_WIDTH_20_NOHT; 306 307 if (link != rsvd_for && 308 rcu_access_pointer(link->conf->chanctx_conf) != &ctx->conf) 309 continue; 310 311 switch (link->sdata->vif.type) { 312 case NL80211_IFTYPE_AP: 313 case NL80211_IFTYPE_AP_VLAN: 314 width = ieee80211_get_max_required_bw(link); 315 break; 316 case NL80211_IFTYPE_STATION: 317 /* 318 * The ap's sta->bandwidth is not set yet at this 319 * point, so take the width from the chandef, but 320 * account also for TDLS peers 321 */ 322 width = max(link->conf->chanreq.oper.width, 323 ieee80211_get_max_required_bw(link)); 324 break; 325 case NL80211_IFTYPE_P2P_DEVICE: 326 case NL80211_IFTYPE_NAN: 327 continue; 328 case NL80211_IFTYPE_ADHOC: 329 case NL80211_IFTYPE_MESH_POINT: 330 case NL80211_IFTYPE_OCB: 331 width = link->conf->chanreq.oper.width; 332 break; 333 case NL80211_IFTYPE_WDS: 334 case NL80211_IFTYPE_UNSPECIFIED: 335 case NUM_NL80211_IFTYPES: 336 case NL80211_IFTYPE_MONITOR: 337 case NL80211_IFTYPE_P2P_CLIENT: 338 case NL80211_IFTYPE_P2P_GO: 339 WARN_ON_ONCE(1); 340 } 341 342 max_bw = max(max_bw, width); 343 } 344 345 /* use the configured bandwidth in case of monitor interface */ 346 sdata = wiphy_dereference(local->hw.wiphy, local->monitor_sdata); 347 if (sdata && 348 rcu_access_pointer(sdata->vif.bss_conf.chanctx_conf) == &ctx->conf) 349 max_bw = max(max_bw, ctx->conf.def.width); 350 351 return max_bw; 352 } 353 354 /* 355 * recalc the min required chan width of the channel context, which is 356 * the max of min required widths of all the interfaces bound to this 357 * channel context. 358 */ 359 static u32 360 _ieee80211_recalc_chanctx_min_def(struct ieee80211_local *local, 361 struct ieee80211_chanctx *ctx, 362 struct ieee80211_link_data *rsvd_for) 363 { 364 enum nl80211_chan_width max_bw; 365 struct cfg80211_chan_def min_def; 366 367 lockdep_assert_wiphy(local->hw.wiphy); 368 369 /* don't optimize non-20MHz based and radar_enabled confs */ 370 if (ctx->conf.def.width == NL80211_CHAN_WIDTH_5 || 371 ctx->conf.def.width == NL80211_CHAN_WIDTH_10 || 372 ctx->conf.def.width == NL80211_CHAN_WIDTH_1 || 373 ctx->conf.def.width == NL80211_CHAN_WIDTH_2 || 374 ctx->conf.def.width == NL80211_CHAN_WIDTH_4 || 375 ctx->conf.def.width == NL80211_CHAN_WIDTH_8 || 376 ctx->conf.def.width == NL80211_CHAN_WIDTH_16 || 377 ctx->conf.radar_enabled) { 378 ctx->conf.min_def = ctx->conf.def; 379 return 0; 380 } 381 382 max_bw = ieee80211_get_chanctx_max_required_bw(local, ctx, rsvd_for); 383 384 /* downgrade chandef up to max_bw */ 385 min_def = ctx->conf.def; 386 while (min_def.width > max_bw) 387 ieee80211_chandef_downgrade(&min_def, NULL); 388 389 if (cfg80211_chandef_identical(&ctx->conf.min_def, &min_def)) 390 return 0; 391 392 ctx->conf.min_def = min_def; 393 if (!ctx->driver_present) 394 return 0; 395 396 return IEEE80211_CHANCTX_CHANGE_MIN_WIDTH; 397 } 398 399 /* calling this function is assuming that station vif is updated to 400 * lates changes by calling ieee80211_link_update_chanreq 401 */ 402 static void ieee80211_chan_bw_change(struct ieee80211_local *local, 403 struct ieee80211_chanctx *ctx, 404 bool narrowed) 405 { 406 struct sta_info *sta; 407 struct ieee80211_supported_band *sband = 408 local->hw.wiphy->bands[ctx->conf.def.chan->band]; 409 410 rcu_read_lock(); 411 list_for_each_entry_rcu(sta, &local->sta_list, 412 list) { 413 struct ieee80211_sub_if_data *sdata = sta->sdata; 414 enum ieee80211_sta_rx_bandwidth new_sta_bw; 415 unsigned int link_id; 416 417 if (!ieee80211_sdata_running(sta->sdata)) 418 continue; 419 420 for (link_id = 0; link_id < ARRAY_SIZE(sta->sdata->link); link_id++) { 421 struct ieee80211_bss_conf *link_conf = 422 rcu_dereference(sdata->vif.link_conf[link_id]); 423 struct link_sta_info *link_sta; 424 425 if (!link_conf) 426 continue; 427 428 if (rcu_access_pointer(link_conf->chanctx_conf) != &ctx->conf) 429 continue; 430 431 link_sta = rcu_dereference(sta->link[link_id]); 432 if (!link_sta) 433 continue; 434 435 new_sta_bw = ieee80211_sta_cur_vht_bw(link_sta); 436 437 /* nothing change */ 438 if (new_sta_bw == link_sta->pub->bandwidth) 439 continue; 440 441 /* vif changed to narrow BW and narrow BW for station wasn't 442 * requested or vise versa */ 443 if ((new_sta_bw < link_sta->pub->bandwidth) == !narrowed) 444 continue; 445 446 link_sta->pub->bandwidth = new_sta_bw; 447 rate_control_rate_update(local, sband, sta, link_id, 448 IEEE80211_RC_BW_CHANGED); 449 } 450 } 451 rcu_read_unlock(); 452 } 453 454 /* 455 * recalc the min required chan width of the channel context, which is 456 * the max of min required widths of all the interfaces bound to this 457 * channel context. 458 */ 459 void ieee80211_recalc_chanctx_min_def(struct ieee80211_local *local, 460 struct ieee80211_chanctx *ctx, 461 struct ieee80211_link_data *rsvd_for) 462 { 463 u32 changed = _ieee80211_recalc_chanctx_min_def(local, ctx, rsvd_for); 464 465 if (!changed) 466 return; 467 468 /* check is BW narrowed */ 469 ieee80211_chan_bw_change(local, ctx, true); 470 471 drv_change_chanctx(local, ctx, changed); 472 473 /* check is BW wider */ 474 ieee80211_chan_bw_change(local, ctx, false); 475 } 476 477 static void _ieee80211_change_chanctx(struct ieee80211_local *local, 478 struct ieee80211_chanctx *ctx, 479 struct ieee80211_chanctx *old_ctx, 480 const struct ieee80211_chan_req *chanreq, 481 struct ieee80211_link_data *rsvd_for) 482 { 483 const struct cfg80211_chan_def *chandef = &chanreq->oper; 484 struct ieee80211_chan_req ctx_req = { 485 .oper = ctx->conf.def, 486 .ap = ctx->conf.ap, 487 }; 488 u32 changed = 0; 489 490 /* expected to handle only 20/40/80/160/320 channel widths */ 491 switch (chandef->width) { 492 case NL80211_CHAN_WIDTH_20_NOHT: 493 case NL80211_CHAN_WIDTH_20: 494 case NL80211_CHAN_WIDTH_40: 495 case NL80211_CHAN_WIDTH_80: 496 case NL80211_CHAN_WIDTH_80P80: 497 case NL80211_CHAN_WIDTH_160: 498 case NL80211_CHAN_WIDTH_320: 499 break; 500 default: 501 WARN_ON(1); 502 } 503 504 /* Check maybe BW narrowed - we do this _before_ calling recalc_chanctx_min_def 505 * due to maybe not returning from it, e.g in case new context was added 506 * first time with all parameters up to date. 507 */ 508 ieee80211_chan_bw_change(local, old_ctx, true); 509 510 if (ieee80211_chanreq_identical(&ctx_req, chanreq)) { 511 ieee80211_recalc_chanctx_min_def(local, ctx, rsvd_for); 512 return; 513 } 514 515 WARN_ON(ieee80211_chanctx_refcount(local, ctx) > 1 && 516 !cfg80211_chandef_compatible(&ctx->conf.def, &chanreq->oper)); 517 518 ieee80211_remove_wbrf(local, &ctx->conf.def); 519 520 if (!cfg80211_chandef_identical(&ctx->conf.def, &chanreq->oper)) { 521 if (ctx->conf.def.width != chanreq->oper.width) 522 changed |= IEEE80211_CHANCTX_CHANGE_WIDTH; 523 if (ctx->conf.def.punctured != chanreq->oper.punctured) 524 changed |= IEEE80211_CHANCTX_CHANGE_PUNCTURING; 525 } 526 if (!cfg80211_chandef_identical(&ctx->conf.ap, &chanreq->ap)) 527 changed |= IEEE80211_CHANCTX_CHANGE_AP; 528 ctx->conf.def = *chandef; 529 ctx->conf.ap = chanreq->ap; 530 531 /* check if min chanctx also changed */ 532 changed |= _ieee80211_recalc_chanctx_min_def(local, ctx, rsvd_for); 533 534 ieee80211_add_wbrf(local, &ctx->conf.def); 535 536 drv_change_chanctx(local, ctx, changed); 537 538 /* check if BW is wider */ 539 ieee80211_chan_bw_change(local, old_ctx, false); 540 } 541 542 static void ieee80211_change_chanctx(struct ieee80211_local *local, 543 struct ieee80211_chanctx *ctx, 544 struct ieee80211_chanctx *old_ctx, 545 const struct ieee80211_chan_req *chanreq) 546 { 547 _ieee80211_change_chanctx(local, ctx, old_ctx, chanreq, NULL); 548 } 549 550 static struct ieee80211_chanctx * 551 ieee80211_find_chanctx(struct ieee80211_local *local, 552 const struct ieee80211_chan_req *chanreq, 553 enum ieee80211_chanctx_mode mode) 554 { 555 struct ieee80211_chan_req tmp; 556 struct ieee80211_chanctx *ctx; 557 558 lockdep_assert_wiphy(local->hw.wiphy); 559 560 if (mode == IEEE80211_CHANCTX_EXCLUSIVE) 561 return NULL; 562 563 list_for_each_entry(ctx, &local->chanctx_list, list) { 564 const struct ieee80211_chan_req *compat; 565 566 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACE_NONE) 567 continue; 568 569 if (ctx->mode == IEEE80211_CHANCTX_EXCLUSIVE) 570 continue; 571 572 compat = ieee80211_chanctx_compatible(ctx, chanreq, &tmp); 573 if (!compat) 574 continue; 575 576 compat = ieee80211_chanctx_reserved_chanreq(local, ctx, 577 compat, &tmp); 578 if (!compat) 579 continue; 580 581 ieee80211_change_chanctx(local, ctx, ctx, compat); 582 583 return ctx; 584 } 585 586 return NULL; 587 } 588 589 bool ieee80211_is_radar_required(struct ieee80211_local *local) 590 { 591 struct ieee80211_link_data *link; 592 593 lockdep_assert_wiphy(local->hw.wiphy); 594 595 for_each_sdata_link(local, link) { 596 if (link->radar_required) 597 return true; 598 } 599 600 return false; 601 } 602 603 static bool 604 ieee80211_chanctx_radar_required(struct ieee80211_local *local, 605 struct ieee80211_chanctx *ctx) 606 { 607 struct ieee80211_chanctx_conf *conf = &ctx->conf; 608 struct ieee80211_link_data *link; 609 610 lockdep_assert_wiphy(local->hw.wiphy); 611 612 for_each_sdata_link(local, link) { 613 if (rcu_access_pointer(link->conf->chanctx_conf) != conf) 614 continue; 615 if (!link->radar_required) 616 continue; 617 return true; 618 } 619 620 return false; 621 } 622 623 static struct ieee80211_chanctx * 624 ieee80211_alloc_chanctx(struct ieee80211_local *local, 625 const struct ieee80211_chan_req *chanreq, 626 enum ieee80211_chanctx_mode mode) 627 { 628 struct ieee80211_chanctx *ctx; 629 630 lockdep_assert_wiphy(local->hw.wiphy); 631 632 ctx = kzalloc(sizeof(*ctx) + local->hw.chanctx_data_size, GFP_KERNEL); 633 if (!ctx) 634 return NULL; 635 636 INIT_LIST_HEAD(&ctx->assigned_links); 637 INIT_LIST_HEAD(&ctx->reserved_links); 638 ctx->conf.def = chanreq->oper; 639 ctx->conf.ap = chanreq->ap; 640 ctx->conf.rx_chains_static = 1; 641 ctx->conf.rx_chains_dynamic = 1; 642 ctx->mode = mode; 643 ctx->conf.radar_enabled = false; 644 _ieee80211_recalc_chanctx_min_def(local, ctx, NULL); 645 646 return ctx; 647 } 648 649 static int ieee80211_add_chanctx(struct ieee80211_local *local, 650 struct ieee80211_chanctx *ctx) 651 { 652 u32 changed; 653 int err; 654 655 lockdep_assert_wiphy(local->hw.wiphy); 656 657 ieee80211_add_wbrf(local, &ctx->conf.def); 658 659 /* turn idle off *before* setting channel -- some drivers need that */ 660 changed = ieee80211_idle_off(local); 661 if (changed) 662 ieee80211_hw_config(local, changed); 663 664 err = drv_add_chanctx(local, ctx); 665 if (err) { 666 ieee80211_recalc_idle(local); 667 return err; 668 } 669 670 return 0; 671 } 672 673 static struct ieee80211_chanctx * 674 ieee80211_new_chanctx(struct ieee80211_local *local, 675 const struct ieee80211_chan_req *chanreq, 676 enum ieee80211_chanctx_mode mode) 677 { 678 struct ieee80211_chanctx *ctx; 679 int err; 680 681 lockdep_assert_wiphy(local->hw.wiphy); 682 683 ctx = ieee80211_alloc_chanctx(local, chanreq, mode); 684 if (!ctx) 685 return ERR_PTR(-ENOMEM); 686 687 err = ieee80211_add_chanctx(local, ctx); 688 if (err) { 689 kfree(ctx); 690 return ERR_PTR(err); 691 } 692 693 list_add_rcu(&ctx->list, &local->chanctx_list); 694 return ctx; 695 } 696 697 static void ieee80211_del_chanctx(struct ieee80211_local *local, 698 struct ieee80211_chanctx *ctx, 699 bool skip_idle_recalc) 700 { 701 lockdep_assert_wiphy(local->hw.wiphy); 702 703 drv_remove_chanctx(local, ctx); 704 705 if (!skip_idle_recalc) 706 ieee80211_recalc_idle(local); 707 708 ieee80211_remove_wbrf(local, &ctx->conf.def); 709 } 710 711 static void ieee80211_free_chanctx(struct ieee80211_local *local, 712 struct ieee80211_chanctx *ctx, 713 bool skip_idle_recalc) 714 { 715 lockdep_assert_wiphy(local->hw.wiphy); 716 717 WARN_ON_ONCE(ieee80211_chanctx_refcount(local, ctx) != 0); 718 719 list_del_rcu(&ctx->list); 720 ieee80211_del_chanctx(local, ctx, skip_idle_recalc); 721 kfree_rcu(ctx, rcu_head); 722 } 723 724 void ieee80211_recalc_chanctx_chantype(struct ieee80211_local *local, 725 struct ieee80211_chanctx *ctx) 726 { 727 struct ieee80211_chanctx_conf *conf = &ctx->conf; 728 const struct ieee80211_chan_req *compat = NULL; 729 struct ieee80211_link_data *link; 730 struct ieee80211_chan_req tmp; 731 struct sta_info *sta; 732 733 lockdep_assert_wiphy(local->hw.wiphy); 734 735 for_each_sdata_link(local, link) { 736 struct ieee80211_bss_conf *link_conf; 737 738 if (link->sdata->vif.type == NL80211_IFTYPE_AP_VLAN) 739 continue; 740 741 link_conf = link->conf; 742 743 if (rcu_access_pointer(link_conf->chanctx_conf) != conf) 744 continue; 745 746 if (!compat) 747 compat = &link_conf->chanreq; 748 749 compat = ieee80211_chanreq_compatible(&link_conf->chanreq, 750 compat, &tmp); 751 if (WARN_ON_ONCE(!compat)) 752 return; 753 } 754 755 if (WARN_ON_ONCE(!compat)) 756 return; 757 758 /* TDLS peers can sometimes affect the chandef width */ 759 list_for_each_entry(sta, &local->sta_list, list) { 760 struct ieee80211_chan_req tdls_chanreq = {}; 761 if (!sta->uploaded || 762 !test_sta_flag(sta, WLAN_STA_TDLS_WIDER_BW) || 763 !test_sta_flag(sta, WLAN_STA_AUTHORIZED) || 764 !sta->tdls_chandef.chan) 765 continue; 766 767 tdls_chanreq.oper = sta->tdls_chandef; 768 769 /* note this always fills and returns &tmp if compat */ 770 compat = ieee80211_chanreq_compatible(&tdls_chanreq, 771 compat, &tmp); 772 if (WARN_ON_ONCE(!compat)) 773 return; 774 } 775 776 ieee80211_change_chanctx(local, ctx, ctx, compat); 777 } 778 779 static void ieee80211_recalc_radar_chanctx(struct ieee80211_local *local, 780 struct ieee80211_chanctx *chanctx) 781 { 782 bool radar_enabled; 783 784 lockdep_assert_wiphy(local->hw.wiphy); 785 786 radar_enabled = ieee80211_chanctx_radar_required(local, chanctx); 787 788 if (radar_enabled == chanctx->conf.radar_enabled) 789 return; 790 791 chanctx->conf.radar_enabled = radar_enabled; 792 793 drv_change_chanctx(local, chanctx, IEEE80211_CHANCTX_CHANGE_RADAR); 794 } 795 796 static int ieee80211_assign_link_chanctx(struct ieee80211_link_data *link, 797 struct ieee80211_chanctx *new_ctx) 798 { 799 struct ieee80211_sub_if_data *sdata = link->sdata; 800 struct ieee80211_local *local = sdata->local; 801 struct ieee80211_chanctx_conf *conf; 802 struct ieee80211_chanctx *curr_ctx = NULL; 803 int ret = 0; 804 805 if (WARN_ON(sdata->vif.type == NL80211_IFTYPE_NAN)) 806 return -EOPNOTSUPP; 807 808 conf = rcu_dereference_protected(link->conf->chanctx_conf, 809 lockdep_is_held(&local->hw.wiphy->mtx)); 810 811 if (conf) { 812 curr_ctx = container_of(conf, struct ieee80211_chanctx, conf); 813 814 drv_unassign_vif_chanctx(local, sdata, link->conf, curr_ctx); 815 conf = NULL; 816 list_del(&link->assigned_chanctx_list); 817 } 818 819 if (new_ctx) { 820 /* recalc considering the link we'll use it for now */ 821 ieee80211_recalc_chanctx_min_def(local, new_ctx, link); 822 823 ret = drv_assign_vif_chanctx(local, sdata, link->conf, new_ctx); 824 if (ret) 825 goto out; 826 827 conf = &new_ctx->conf; 828 list_add(&link->assigned_chanctx_list, 829 &new_ctx->assigned_links); 830 } 831 832 out: 833 rcu_assign_pointer(link->conf->chanctx_conf, conf); 834 835 sdata->vif.cfg.idle = !conf; 836 837 if (curr_ctx && ieee80211_chanctx_num_assigned(local, curr_ctx) > 0) { 838 ieee80211_recalc_chanctx_chantype(local, curr_ctx); 839 ieee80211_recalc_smps_chanctx(local, curr_ctx); 840 ieee80211_recalc_radar_chanctx(local, curr_ctx); 841 ieee80211_recalc_chanctx_min_def(local, curr_ctx, NULL); 842 } 843 844 if (new_ctx && ieee80211_chanctx_num_assigned(local, new_ctx) > 0) { 845 ieee80211_recalc_txpower(sdata, false); 846 ieee80211_recalc_chanctx_min_def(local, new_ctx, NULL); 847 } 848 849 if (sdata->vif.type != NL80211_IFTYPE_P2P_DEVICE && 850 sdata->vif.type != NL80211_IFTYPE_MONITOR) 851 ieee80211_vif_cfg_change_notify(sdata, BSS_CHANGED_IDLE); 852 853 ieee80211_check_fast_xmit_iface(sdata); 854 855 return ret; 856 } 857 858 void ieee80211_recalc_smps_chanctx(struct ieee80211_local *local, 859 struct ieee80211_chanctx *chanctx) 860 { 861 struct ieee80211_sub_if_data *sdata; 862 u8 rx_chains_static, rx_chains_dynamic; 863 struct ieee80211_link_data *link; 864 865 lockdep_assert_wiphy(local->hw.wiphy); 866 867 rx_chains_static = 1; 868 rx_chains_dynamic = 1; 869 870 for_each_sdata_link(local, link) { 871 u8 needed_static, needed_dynamic; 872 873 switch (link->sdata->vif.type) { 874 case NL80211_IFTYPE_STATION: 875 if (!link->sdata->u.mgd.associated) 876 continue; 877 break; 878 case NL80211_IFTYPE_AP: 879 case NL80211_IFTYPE_ADHOC: 880 case NL80211_IFTYPE_MESH_POINT: 881 case NL80211_IFTYPE_OCB: 882 break; 883 default: 884 continue; 885 } 886 887 if (rcu_access_pointer(link->conf->chanctx_conf) != &chanctx->conf) 888 continue; 889 890 switch (link->smps_mode) { 891 default: 892 WARN_ONCE(1, "Invalid SMPS mode %d\n", 893 link->smps_mode); 894 fallthrough; 895 case IEEE80211_SMPS_OFF: 896 needed_static = link->needed_rx_chains; 897 needed_dynamic = link->needed_rx_chains; 898 break; 899 case IEEE80211_SMPS_DYNAMIC: 900 needed_static = 1; 901 needed_dynamic = link->needed_rx_chains; 902 break; 903 case IEEE80211_SMPS_STATIC: 904 needed_static = 1; 905 needed_dynamic = 1; 906 break; 907 } 908 909 rx_chains_static = max(rx_chains_static, needed_static); 910 rx_chains_dynamic = max(rx_chains_dynamic, needed_dynamic); 911 } 912 913 /* Disable SMPS for the monitor interface */ 914 sdata = wiphy_dereference(local->hw.wiphy, local->monitor_sdata); 915 if (sdata && 916 rcu_access_pointer(sdata->vif.bss_conf.chanctx_conf) == &chanctx->conf) 917 rx_chains_dynamic = rx_chains_static = local->rx_chains; 918 919 if (rx_chains_static == chanctx->conf.rx_chains_static && 920 rx_chains_dynamic == chanctx->conf.rx_chains_dynamic) 921 return; 922 923 chanctx->conf.rx_chains_static = rx_chains_static; 924 chanctx->conf.rx_chains_dynamic = rx_chains_dynamic; 925 drv_change_chanctx(local, chanctx, IEEE80211_CHANCTX_CHANGE_RX_CHAINS); 926 } 927 928 static void 929 __ieee80211_link_copy_chanctx_to_vlans(struct ieee80211_link_data *link, 930 bool clear) 931 { 932 struct ieee80211_sub_if_data *sdata = link->sdata; 933 unsigned int link_id = link->link_id; 934 struct ieee80211_bss_conf *link_conf = link->conf; 935 struct ieee80211_local *local __maybe_unused = sdata->local; 936 struct ieee80211_sub_if_data *vlan; 937 struct ieee80211_chanctx_conf *conf; 938 939 if (WARN_ON(sdata->vif.type != NL80211_IFTYPE_AP)) 940 return; 941 942 lockdep_assert_wiphy(local->hw.wiphy); 943 944 /* Check that conf exists, even when clearing this function 945 * must be called with the AP's channel context still there 946 * as it would otherwise cause VLANs to have an invalid 947 * channel context pointer for a while, possibly pointing 948 * to a channel context that has already been freed. 949 */ 950 conf = rcu_dereference_protected(link_conf->chanctx_conf, 951 lockdep_is_held(&local->hw.wiphy->mtx)); 952 WARN_ON(!conf); 953 954 if (clear) 955 conf = NULL; 956 957 list_for_each_entry(vlan, &sdata->u.ap.vlans, u.vlan.list) { 958 struct ieee80211_bss_conf *vlan_conf; 959 960 vlan_conf = wiphy_dereference(local->hw.wiphy, 961 vlan->vif.link_conf[link_id]); 962 if (WARN_ON(!vlan_conf)) 963 continue; 964 965 rcu_assign_pointer(vlan_conf->chanctx_conf, conf); 966 } 967 } 968 969 void ieee80211_link_copy_chanctx_to_vlans(struct ieee80211_link_data *link, 970 bool clear) 971 { 972 struct ieee80211_local *local = link->sdata->local; 973 974 lockdep_assert_wiphy(local->hw.wiphy); 975 976 __ieee80211_link_copy_chanctx_to_vlans(link, clear); 977 } 978 979 int ieee80211_link_unreserve_chanctx(struct ieee80211_link_data *link) 980 { 981 struct ieee80211_sub_if_data *sdata = link->sdata; 982 struct ieee80211_chanctx *ctx = link->reserved_chanctx; 983 984 lockdep_assert_wiphy(sdata->local->hw.wiphy); 985 986 if (WARN_ON(!ctx)) 987 return -EINVAL; 988 989 list_del(&link->reserved_chanctx_list); 990 link->reserved_chanctx = NULL; 991 992 if (ieee80211_chanctx_refcount(sdata->local, ctx) == 0) { 993 if (ctx->replace_state == IEEE80211_CHANCTX_REPLACES_OTHER) { 994 if (WARN_ON(!ctx->replace_ctx)) 995 return -EINVAL; 996 997 WARN_ON(ctx->replace_ctx->replace_state != 998 IEEE80211_CHANCTX_WILL_BE_REPLACED); 999 WARN_ON(ctx->replace_ctx->replace_ctx != ctx); 1000 1001 ctx->replace_ctx->replace_ctx = NULL; 1002 ctx->replace_ctx->replace_state = 1003 IEEE80211_CHANCTX_REPLACE_NONE; 1004 1005 list_del_rcu(&ctx->list); 1006 kfree_rcu(ctx, rcu_head); 1007 } else { 1008 ieee80211_free_chanctx(sdata->local, ctx, false); 1009 } 1010 } 1011 1012 return 0; 1013 } 1014 1015 int ieee80211_link_reserve_chanctx(struct ieee80211_link_data *link, 1016 const struct ieee80211_chan_req *chanreq, 1017 enum ieee80211_chanctx_mode mode, 1018 bool radar_required) 1019 { 1020 struct ieee80211_sub_if_data *sdata = link->sdata; 1021 struct ieee80211_local *local = sdata->local; 1022 struct ieee80211_chanctx *new_ctx, *curr_ctx, *ctx; 1023 1024 lockdep_assert_wiphy(local->hw.wiphy); 1025 1026 curr_ctx = ieee80211_link_get_chanctx(link); 1027 if (curr_ctx && !local->ops->switch_vif_chanctx) 1028 return -EOPNOTSUPP; 1029 1030 new_ctx = ieee80211_find_reservation_chanctx(local, chanreq, mode); 1031 if (!new_ctx) { 1032 if (ieee80211_can_create_new_chanctx(local)) { 1033 new_ctx = ieee80211_new_chanctx(local, chanreq, mode); 1034 if (IS_ERR(new_ctx)) 1035 return PTR_ERR(new_ctx); 1036 } else { 1037 if (!curr_ctx || 1038 (curr_ctx->replace_state == 1039 IEEE80211_CHANCTX_WILL_BE_REPLACED) || 1040 !list_empty(&curr_ctx->reserved_links)) { 1041 /* 1042 * Another link already requested this context 1043 * for a reservation. Find another one hoping 1044 * all links assigned to it will also switch 1045 * soon enough. 1046 * 1047 * TODO: This needs a little more work as some 1048 * cases (more than 2 chanctx capable devices) 1049 * may fail which could otherwise succeed 1050 * provided some channel context juggling was 1051 * performed. 1052 * 1053 * Consider ctx1..3, link1..6, each ctx has 2 1054 * links. link1 and link2 from ctx1 request new 1055 * different chandefs starting 2 in-place 1056 * reserations with ctx4 and ctx5 replacing 1057 * ctx1 and ctx2 respectively. Next link5 and 1058 * link6 from ctx3 reserve ctx4. If link3 and 1059 * link4 remain on ctx2 as they are then this 1060 * fails unless `replace_ctx` from ctx5 is 1061 * replaced with ctx3. 1062 */ 1063 list_for_each_entry(ctx, &local->chanctx_list, 1064 list) { 1065 if (ctx->replace_state != 1066 IEEE80211_CHANCTX_REPLACE_NONE) 1067 continue; 1068 1069 if (!list_empty(&ctx->reserved_links)) 1070 continue; 1071 1072 curr_ctx = ctx; 1073 break; 1074 } 1075 } 1076 1077 /* 1078 * If that's true then all available contexts already 1079 * have reservations and cannot be used. 1080 */ 1081 if (!curr_ctx || 1082 (curr_ctx->replace_state == 1083 IEEE80211_CHANCTX_WILL_BE_REPLACED) || 1084 !list_empty(&curr_ctx->reserved_links)) 1085 return -EBUSY; 1086 1087 new_ctx = ieee80211_alloc_chanctx(local, chanreq, mode); 1088 if (!new_ctx) 1089 return -ENOMEM; 1090 1091 new_ctx->replace_ctx = curr_ctx; 1092 new_ctx->replace_state = 1093 IEEE80211_CHANCTX_REPLACES_OTHER; 1094 1095 curr_ctx->replace_ctx = new_ctx; 1096 curr_ctx->replace_state = 1097 IEEE80211_CHANCTX_WILL_BE_REPLACED; 1098 1099 list_add_rcu(&new_ctx->list, &local->chanctx_list); 1100 } 1101 } 1102 1103 list_add(&link->reserved_chanctx_list, &new_ctx->reserved_links); 1104 link->reserved_chanctx = new_ctx; 1105 link->reserved = *chanreq; 1106 link->reserved_radar_required = radar_required; 1107 link->reserved_ready = false; 1108 1109 return 0; 1110 } 1111 1112 static void 1113 ieee80211_link_chanctx_reservation_complete(struct ieee80211_link_data *link) 1114 { 1115 struct ieee80211_sub_if_data *sdata = link->sdata; 1116 1117 switch (sdata->vif.type) { 1118 case NL80211_IFTYPE_ADHOC: 1119 case NL80211_IFTYPE_AP: 1120 case NL80211_IFTYPE_MESH_POINT: 1121 case NL80211_IFTYPE_OCB: 1122 wiphy_work_queue(sdata->local->hw.wiphy, 1123 &link->csa_finalize_work); 1124 break; 1125 case NL80211_IFTYPE_STATION: 1126 wiphy_delayed_work_queue(sdata->local->hw.wiphy, 1127 &link->u.mgd.chswitch_work, 0); 1128 break; 1129 case NL80211_IFTYPE_UNSPECIFIED: 1130 case NL80211_IFTYPE_AP_VLAN: 1131 case NL80211_IFTYPE_WDS: 1132 case NL80211_IFTYPE_MONITOR: 1133 case NL80211_IFTYPE_P2P_CLIENT: 1134 case NL80211_IFTYPE_P2P_GO: 1135 case NL80211_IFTYPE_P2P_DEVICE: 1136 case NL80211_IFTYPE_NAN: 1137 case NUM_NL80211_IFTYPES: 1138 WARN_ON(1); 1139 break; 1140 } 1141 } 1142 1143 static void 1144 ieee80211_link_update_chanreq(struct ieee80211_link_data *link, 1145 const struct ieee80211_chan_req *chanreq) 1146 { 1147 struct ieee80211_sub_if_data *sdata = link->sdata; 1148 unsigned int link_id = link->link_id; 1149 struct ieee80211_sub_if_data *vlan; 1150 1151 link->conf->chanreq = *chanreq; 1152 1153 if (sdata->vif.type != NL80211_IFTYPE_AP) 1154 return; 1155 1156 list_for_each_entry(vlan, &sdata->u.ap.vlans, u.vlan.list) { 1157 struct ieee80211_bss_conf *vlan_conf; 1158 1159 vlan_conf = wiphy_dereference(sdata->local->hw.wiphy, 1160 vlan->vif.link_conf[link_id]); 1161 if (WARN_ON(!vlan_conf)) 1162 continue; 1163 1164 vlan_conf->chanreq = *chanreq; 1165 } 1166 } 1167 1168 static int 1169 ieee80211_link_use_reserved_reassign(struct ieee80211_link_data *link) 1170 { 1171 struct ieee80211_sub_if_data *sdata = link->sdata; 1172 struct ieee80211_bss_conf *link_conf = link->conf; 1173 struct ieee80211_local *local = sdata->local; 1174 struct ieee80211_vif_chanctx_switch vif_chsw[1] = {}; 1175 struct ieee80211_chanctx *old_ctx, *new_ctx; 1176 const struct ieee80211_chan_req *chanreq; 1177 struct ieee80211_chan_req tmp; 1178 u64 changed = 0; 1179 int err; 1180 1181 lockdep_assert_wiphy(local->hw.wiphy); 1182 1183 new_ctx = link->reserved_chanctx; 1184 old_ctx = ieee80211_link_get_chanctx(link); 1185 1186 if (WARN_ON(!link->reserved_ready)) 1187 return -EBUSY; 1188 1189 if (WARN_ON(!new_ctx)) 1190 return -EINVAL; 1191 1192 if (WARN_ON(!old_ctx)) 1193 return -EINVAL; 1194 1195 if (WARN_ON(new_ctx->replace_state == 1196 IEEE80211_CHANCTX_REPLACES_OTHER)) 1197 return -EINVAL; 1198 1199 chanreq = ieee80211_chanctx_non_reserved_chandef(local, new_ctx, 1200 &link->reserved, 1201 &tmp); 1202 if (WARN_ON(!chanreq)) 1203 return -EINVAL; 1204 1205 if (link_conf->chanreq.oper.width != link->reserved.oper.width) 1206 changed = BSS_CHANGED_BANDWIDTH; 1207 1208 ieee80211_link_update_chanreq(link, &link->reserved); 1209 1210 _ieee80211_change_chanctx(local, new_ctx, old_ctx, chanreq, link); 1211 1212 vif_chsw[0].vif = &sdata->vif; 1213 vif_chsw[0].old_ctx = &old_ctx->conf; 1214 vif_chsw[0].new_ctx = &new_ctx->conf; 1215 vif_chsw[0].link_conf = link->conf; 1216 1217 list_del(&link->reserved_chanctx_list); 1218 link->reserved_chanctx = NULL; 1219 1220 err = drv_switch_vif_chanctx(local, vif_chsw, 1, 1221 CHANCTX_SWMODE_REASSIGN_VIF); 1222 if (err) { 1223 if (ieee80211_chanctx_refcount(local, new_ctx) == 0) 1224 ieee80211_free_chanctx(local, new_ctx, false); 1225 1226 goto out; 1227 } 1228 1229 list_move(&link->assigned_chanctx_list, &new_ctx->assigned_links); 1230 rcu_assign_pointer(link_conf->chanctx_conf, &new_ctx->conf); 1231 1232 if (sdata->vif.type == NL80211_IFTYPE_AP) 1233 __ieee80211_link_copy_chanctx_to_vlans(link, false); 1234 1235 ieee80211_check_fast_xmit_iface(sdata); 1236 1237 if (ieee80211_chanctx_refcount(local, old_ctx) == 0) 1238 ieee80211_free_chanctx(local, old_ctx, false); 1239 1240 ieee80211_recalc_chanctx_min_def(local, new_ctx, NULL); 1241 ieee80211_recalc_smps_chanctx(local, new_ctx); 1242 ieee80211_recalc_radar_chanctx(local, new_ctx); 1243 1244 if (changed) 1245 ieee80211_link_info_change_notify(sdata, link, changed); 1246 1247 out: 1248 ieee80211_link_chanctx_reservation_complete(link); 1249 return err; 1250 } 1251 1252 static int 1253 ieee80211_link_use_reserved_assign(struct ieee80211_link_data *link) 1254 { 1255 struct ieee80211_sub_if_data *sdata = link->sdata; 1256 struct ieee80211_local *local = sdata->local; 1257 struct ieee80211_chanctx *old_ctx, *new_ctx; 1258 const struct ieee80211_chan_req *chanreq; 1259 struct ieee80211_chan_req tmp; 1260 int err; 1261 1262 old_ctx = ieee80211_link_get_chanctx(link); 1263 new_ctx = link->reserved_chanctx; 1264 1265 if (WARN_ON(!link->reserved_ready)) 1266 return -EINVAL; 1267 1268 if (WARN_ON(old_ctx)) 1269 return -EINVAL; 1270 1271 if (WARN_ON(!new_ctx)) 1272 return -EINVAL; 1273 1274 if (WARN_ON(new_ctx->replace_state == 1275 IEEE80211_CHANCTX_REPLACES_OTHER)) 1276 return -EINVAL; 1277 1278 chanreq = ieee80211_chanctx_non_reserved_chandef(local, new_ctx, 1279 &link->reserved, 1280 &tmp); 1281 if (WARN_ON(!chanreq)) 1282 return -EINVAL; 1283 1284 ieee80211_change_chanctx(local, new_ctx, new_ctx, chanreq); 1285 1286 list_del(&link->reserved_chanctx_list); 1287 link->reserved_chanctx = NULL; 1288 1289 err = ieee80211_assign_link_chanctx(link, new_ctx); 1290 if (err) { 1291 if (ieee80211_chanctx_refcount(local, new_ctx) == 0) 1292 ieee80211_free_chanctx(local, new_ctx, false); 1293 1294 goto out; 1295 } 1296 1297 out: 1298 ieee80211_link_chanctx_reservation_complete(link); 1299 return err; 1300 } 1301 1302 static bool 1303 ieee80211_link_has_in_place_reservation(struct ieee80211_link_data *link) 1304 { 1305 struct ieee80211_sub_if_data *sdata = link->sdata; 1306 struct ieee80211_chanctx *old_ctx, *new_ctx; 1307 1308 lockdep_assert_wiphy(sdata->local->hw.wiphy); 1309 1310 new_ctx = link->reserved_chanctx; 1311 old_ctx = ieee80211_link_get_chanctx(link); 1312 1313 if (!old_ctx) 1314 return false; 1315 1316 if (WARN_ON(!new_ctx)) 1317 return false; 1318 1319 if (old_ctx->replace_state != IEEE80211_CHANCTX_WILL_BE_REPLACED) 1320 return false; 1321 1322 if (new_ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1323 return false; 1324 1325 return true; 1326 } 1327 1328 static int ieee80211_chsw_switch_vifs(struct ieee80211_local *local, 1329 int n_vifs) 1330 { 1331 struct ieee80211_vif_chanctx_switch *vif_chsw; 1332 struct ieee80211_link_data *link; 1333 struct ieee80211_chanctx *ctx, *old_ctx; 1334 int i, err; 1335 1336 lockdep_assert_wiphy(local->hw.wiphy); 1337 1338 vif_chsw = kcalloc(n_vifs, sizeof(vif_chsw[0]), GFP_KERNEL); 1339 if (!vif_chsw) 1340 return -ENOMEM; 1341 1342 i = 0; 1343 list_for_each_entry(ctx, &local->chanctx_list, list) { 1344 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1345 continue; 1346 1347 if (WARN_ON(!ctx->replace_ctx)) { 1348 err = -EINVAL; 1349 goto out; 1350 } 1351 1352 list_for_each_entry(link, &ctx->reserved_links, 1353 reserved_chanctx_list) { 1354 if (!ieee80211_link_has_in_place_reservation(link)) 1355 continue; 1356 1357 old_ctx = ieee80211_link_get_chanctx(link); 1358 vif_chsw[i].vif = &link->sdata->vif; 1359 vif_chsw[i].old_ctx = &old_ctx->conf; 1360 vif_chsw[i].new_ctx = &ctx->conf; 1361 vif_chsw[i].link_conf = link->conf; 1362 1363 i++; 1364 } 1365 } 1366 1367 err = drv_switch_vif_chanctx(local, vif_chsw, n_vifs, 1368 CHANCTX_SWMODE_SWAP_CONTEXTS); 1369 1370 out: 1371 kfree(vif_chsw); 1372 return err; 1373 } 1374 1375 static int ieee80211_chsw_switch_ctxs(struct ieee80211_local *local) 1376 { 1377 struct ieee80211_chanctx *ctx; 1378 int err; 1379 1380 lockdep_assert_wiphy(local->hw.wiphy); 1381 1382 list_for_each_entry(ctx, &local->chanctx_list, list) { 1383 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1384 continue; 1385 1386 if (!list_empty(&ctx->replace_ctx->assigned_links)) 1387 continue; 1388 1389 ieee80211_del_chanctx(local, ctx->replace_ctx, false); 1390 err = ieee80211_add_chanctx(local, ctx); 1391 if (err) 1392 goto err; 1393 } 1394 1395 return 0; 1396 1397 err: 1398 WARN_ON(ieee80211_add_chanctx(local, ctx)); 1399 list_for_each_entry_continue_reverse(ctx, &local->chanctx_list, list) { 1400 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1401 continue; 1402 1403 if (!list_empty(&ctx->replace_ctx->assigned_links)) 1404 continue; 1405 1406 ieee80211_del_chanctx(local, ctx, false); 1407 WARN_ON(ieee80211_add_chanctx(local, ctx->replace_ctx)); 1408 } 1409 1410 return err; 1411 } 1412 1413 static int ieee80211_vif_use_reserved_switch(struct ieee80211_local *local) 1414 { 1415 struct ieee80211_chanctx *ctx, *ctx_tmp, *old_ctx; 1416 int err, n_assigned, n_reserved, n_ready; 1417 int n_ctx = 0, n_vifs_switch = 0, n_vifs_assign = 0, n_vifs_ctxless = 0; 1418 1419 lockdep_assert_wiphy(local->hw.wiphy); 1420 1421 /* 1422 * If there are 2 independent pairs of channel contexts performing 1423 * cross-switch of their vifs this code will still wait until both are 1424 * ready even though it could be possible to switch one before the 1425 * other is ready. 1426 * 1427 * For practical reasons and code simplicity just do a single huge 1428 * switch. 1429 */ 1430 1431 /* 1432 * Verify if the reservation is still feasible. 1433 * - if it's not then disconnect 1434 * - if it is but not all vifs necessary are ready then defer 1435 */ 1436 1437 list_for_each_entry(ctx, &local->chanctx_list, list) { 1438 struct ieee80211_link_data *link; 1439 1440 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1441 continue; 1442 1443 if (WARN_ON(!ctx->replace_ctx)) { 1444 err = -EINVAL; 1445 goto err; 1446 } 1447 1448 n_ctx++; 1449 1450 n_assigned = 0; 1451 n_reserved = 0; 1452 n_ready = 0; 1453 1454 list_for_each_entry(link, &ctx->replace_ctx->assigned_links, 1455 assigned_chanctx_list) { 1456 n_assigned++; 1457 if (link->reserved_chanctx) { 1458 n_reserved++; 1459 if (link->reserved_ready) 1460 n_ready++; 1461 } 1462 } 1463 1464 if (n_assigned != n_reserved) { 1465 if (n_ready == n_reserved) { 1466 wiphy_info(local->hw.wiphy, 1467 "channel context reservation cannot be finalized because some interfaces aren't switching\n"); 1468 err = -EBUSY; 1469 goto err; 1470 } 1471 1472 return -EAGAIN; 1473 } 1474 1475 ctx->conf.radar_enabled = false; 1476 list_for_each_entry(link, &ctx->reserved_links, 1477 reserved_chanctx_list) { 1478 if (ieee80211_link_has_in_place_reservation(link) && 1479 !link->reserved_ready) 1480 return -EAGAIN; 1481 1482 old_ctx = ieee80211_link_get_chanctx(link); 1483 if (old_ctx) { 1484 if (old_ctx->replace_state == 1485 IEEE80211_CHANCTX_WILL_BE_REPLACED) 1486 n_vifs_switch++; 1487 else 1488 n_vifs_assign++; 1489 } else { 1490 n_vifs_ctxless++; 1491 } 1492 1493 if (link->reserved_radar_required) 1494 ctx->conf.radar_enabled = true; 1495 } 1496 } 1497 1498 if (WARN_ON(n_ctx == 0) || 1499 WARN_ON(n_vifs_switch == 0 && 1500 n_vifs_assign == 0 && 1501 n_vifs_ctxless == 0)) { 1502 err = -EINVAL; 1503 goto err; 1504 } 1505 1506 /* 1507 * All necessary vifs are ready. Perform the switch now depending on 1508 * reservations and driver capabilities. 1509 */ 1510 1511 if (n_vifs_switch > 0) { 1512 err = ieee80211_chsw_switch_vifs(local, n_vifs_switch); 1513 if (err) 1514 goto err; 1515 } 1516 1517 if (n_vifs_assign > 0 || n_vifs_ctxless > 0) { 1518 err = ieee80211_chsw_switch_ctxs(local); 1519 if (err) 1520 goto err; 1521 } 1522 1523 /* 1524 * Update all structures, values and pointers to point to new channel 1525 * context(s). 1526 */ 1527 list_for_each_entry(ctx, &local->chanctx_list, list) { 1528 struct ieee80211_link_data *link, *link_tmp; 1529 1530 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1531 continue; 1532 1533 if (WARN_ON(!ctx->replace_ctx)) { 1534 err = -EINVAL; 1535 goto err; 1536 } 1537 1538 list_for_each_entry(link, &ctx->reserved_links, 1539 reserved_chanctx_list) { 1540 struct ieee80211_sub_if_data *sdata = link->sdata; 1541 struct ieee80211_bss_conf *link_conf = link->conf; 1542 u64 changed = 0; 1543 1544 if (!ieee80211_link_has_in_place_reservation(link)) 1545 continue; 1546 1547 rcu_assign_pointer(link_conf->chanctx_conf, 1548 &ctx->conf); 1549 1550 if (sdata->vif.type == NL80211_IFTYPE_AP) 1551 __ieee80211_link_copy_chanctx_to_vlans(link, 1552 false); 1553 1554 ieee80211_check_fast_xmit_iface(sdata); 1555 1556 link->radar_required = link->reserved_radar_required; 1557 1558 if (link_conf->chanreq.oper.width != link->reserved.oper.width) 1559 changed = BSS_CHANGED_BANDWIDTH; 1560 1561 ieee80211_link_update_chanreq(link, &link->reserved); 1562 if (changed) 1563 ieee80211_link_info_change_notify(sdata, 1564 link, 1565 changed); 1566 1567 ieee80211_recalc_txpower(sdata, false); 1568 } 1569 1570 ieee80211_recalc_chanctx_chantype(local, ctx); 1571 ieee80211_recalc_smps_chanctx(local, ctx); 1572 ieee80211_recalc_radar_chanctx(local, ctx); 1573 ieee80211_recalc_chanctx_min_def(local, ctx, NULL); 1574 1575 list_for_each_entry_safe(link, link_tmp, &ctx->reserved_links, 1576 reserved_chanctx_list) { 1577 if (ieee80211_link_get_chanctx(link) != ctx) 1578 continue; 1579 1580 list_del(&link->reserved_chanctx_list); 1581 list_move(&link->assigned_chanctx_list, 1582 &ctx->assigned_links); 1583 link->reserved_chanctx = NULL; 1584 1585 ieee80211_link_chanctx_reservation_complete(link); 1586 } 1587 1588 /* 1589 * This context might have been a dependency for an already 1590 * ready re-assign reservation interface that was deferred. Do 1591 * not propagate error to the caller though. The in-place 1592 * reservation for originally requested interface has already 1593 * succeeded at this point. 1594 */ 1595 list_for_each_entry_safe(link, link_tmp, &ctx->reserved_links, 1596 reserved_chanctx_list) { 1597 if (WARN_ON(ieee80211_link_has_in_place_reservation(link))) 1598 continue; 1599 1600 if (WARN_ON(link->reserved_chanctx != ctx)) 1601 continue; 1602 1603 if (!link->reserved_ready) 1604 continue; 1605 1606 if (ieee80211_link_get_chanctx(link)) 1607 err = ieee80211_link_use_reserved_reassign(link); 1608 else 1609 err = ieee80211_link_use_reserved_assign(link); 1610 1611 if (err) { 1612 link_info(link, 1613 "failed to finalize (re-)assign reservation (err=%d)\n", 1614 err); 1615 ieee80211_link_unreserve_chanctx(link); 1616 cfg80211_stop_iface(local->hw.wiphy, 1617 &link->sdata->wdev, 1618 GFP_KERNEL); 1619 } 1620 } 1621 } 1622 1623 /* 1624 * Finally free old contexts 1625 */ 1626 1627 list_for_each_entry_safe(ctx, ctx_tmp, &local->chanctx_list, list) { 1628 if (ctx->replace_state != IEEE80211_CHANCTX_WILL_BE_REPLACED) 1629 continue; 1630 1631 ctx->replace_ctx->replace_ctx = NULL; 1632 ctx->replace_ctx->replace_state = 1633 IEEE80211_CHANCTX_REPLACE_NONE; 1634 1635 list_del_rcu(&ctx->list); 1636 kfree_rcu(ctx, rcu_head); 1637 } 1638 1639 return 0; 1640 1641 err: 1642 list_for_each_entry(ctx, &local->chanctx_list, list) { 1643 struct ieee80211_link_data *link, *link_tmp; 1644 1645 if (ctx->replace_state != IEEE80211_CHANCTX_REPLACES_OTHER) 1646 continue; 1647 1648 list_for_each_entry_safe(link, link_tmp, &ctx->reserved_links, 1649 reserved_chanctx_list) { 1650 ieee80211_link_unreserve_chanctx(link); 1651 ieee80211_link_chanctx_reservation_complete(link); 1652 } 1653 } 1654 1655 return err; 1656 } 1657 1658 void __ieee80211_link_release_channel(struct ieee80211_link_data *link, 1659 bool skip_idle_recalc) 1660 { 1661 struct ieee80211_sub_if_data *sdata = link->sdata; 1662 struct ieee80211_bss_conf *link_conf = link->conf; 1663 struct ieee80211_local *local = sdata->local; 1664 struct ieee80211_chanctx_conf *conf; 1665 struct ieee80211_chanctx *ctx; 1666 bool use_reserved_switch = false; 1667 1668 lockdep_assert_wiphy(local->hw.wiphy); 1669 1670 conf = rcu_dereference_protected(link_conf->chanctx_conf, 1671 lockdep_is_held(&local->hw.wiphy->mtx)); 1672 if (!conf) 1673 return; 1674 1675 ctx = container_of(conf, struct ieee80211_chanctx, conf); 1676 1677 if (link->reserved_chanctx) { 1678 if (link->reserved_chanctx->replace_state == IEEE80211_CHANCTX_REPLACES_OTHER && 1679 ieee80211_chanctx_num_reserved(local, link->reserved_chanctx) > 1) 1680 use_reserved_switch = true; 1681 1682 ieee80211_link_unreserve_chanctx(link); 1683 } 1684 1685 ieee80211_assign_link_chanctx(link, NULL); 1686 if (ieee80211_chanctx_refcount(local, ctx) == 0) 1687 ieee80211_free_chanctx(local, ctx, skip_idle_recalc); 1688 1689 link->radar_required = false; 1690 1691 /* Unreserving may ready an in-place reservation. */ 1692 if (use_reserved_switch) 1693 ieee80211_vif_use_reserved_switch(local); 1694 } 1695 1696 int ieee80211_link_use_channel(struct ieee80211_link_data *link, 1697 const struct ieee80211_chan_req *chanreq, 1698 enum ieee80211_chanctx_mode mode) 1699 { 1700 struct ieee80211_sub_if_data *sdata = link->sdata; 1701 struct ieee80211_local *local = sdata->local; 1702 struct ieee80211_chanctx *ctx; 1703 u8 radar_detect_width = 0; 1704 int ret; 1705 1706 lockdep_assert_wiphy(local->hw.wiphy); 1707 1708 if (!ieee80211_vif_link_active(&sdata->vif, link->link_id)) { 1709 ieee80211_link_update_chanreq(link, chanreq); 1710 return 0; 1711 } 1712 1713 ret = cfg80211_chandef_dfs_required(local->hw.wiphy, 1714 &chanreq->oper, 1715 sdata->wdev.iftype); 1716 if (ret < 0) 1717 goto out; 1718 if (ret > 0) 1719 radar_detect_width = BIT(chanreq->oper.width); 1720 1721 link->radar_required = ret; 1722 1723 ret = ieee80211_check_combinations(sdata, &chanreq->oper, mode, 1724 radar_detect_width); 1725 if (ret < 0) 1726 goto out; 1727 1728 __ieee80211_link_release_channel(link, false); 1729 1730 ctx = ieee80211_find_chanctx(local, chanreq, mode); 1731 if (!ctx) 1732 ctx = ieee80211_new_chanctx(local, chanreq, mode); 1733 if (IS_ERR(ctx)) { 1734 ret = PTR_ERR(ctx); 1735 goto out; 1736 } 1737 1738 ieee80211_link_update_chanreq(link, chanreq); 1739 1740 ret = ieee80211_assign_link_chanctx(link, ctx); 1741 if (ret) { 1742 /* if assign fails refcount stays the same */ 1743 if (ieee80211_chanctx_refcount(local, ctx) == 0) 1744 ieee80211_free_chanctx(local, ctx, false); 1745 goto out; 1746 } 1747 1748 ieee80211_recalc_smps_chanctx(local, ctx); 1749 ieee80211_recalc_radar_chanctx(local, ctx); 1750 out: 1751 if (ret) 1752 link->radar_required = false; 1753 1754 return ret; 1755 } 1756 1757 int ieee80211_link_use_reserved_context(struct ieee80211_link_data *link) 1758 { 1759 struct ieee80211_sub_if_data *sdata = link->sdata; 1760 struct ieee80211_local *local = sdata->local; 1761 struct ieee80211_chanctx *new_ctx; 1762 struct ieee80211_chanctx *old_ctx; 1763 int err; 1764 1765 lockdep_assert_wiphy(local->hw.wiphy); 1766 1767 new_ctx = link->reserved_chanctx; 1768 old_ctx = ieee80211_link_get_chanctx(link); 1769 1770 if (WARN_ON(!new_ctx)) 1771 return -EINVAL; 1772 1773 if (WARN_ON(new_ctx->replace_state == 1774 IEEE80211_CHANCTX_WILL_BE_REPLACED)) 1775 return -EINVAL; 1776 1777 if (WARN_ON(link->reserved_ready)) 1778 return -EINVAL; 1779 1780 link->reserved_ready = true; 1781 1782 if (new_ctx->replace_state == IEEE80211_CHANCTX_REPLACE_NONE) { 1783 if (old_ctx) 1784 return ieee80211_link_use_reserved_reassign(link); 1785 1786 return ieee80211_link_use_reserved_assign(link); 1787 } 1788 1789 /* 1790 * In-place reservation may need to be finalized now either if: 1791 * a) sdata is taking part in the swapping itself and is the last one 1792 * b) sdata has switched with a re-assign reservation to an existing 1793 * context readying in-place switching of old_ctx 1794 * 1795 * In case of (b) do not propagate the error up because the requested 1796 * sdata already switched successfully. Just spill an extra warning. 1797 * The ieee80211_vif_use_reserved_switch() already stops all necessary 1798 * interfaces upon failure. 1799 */ 1800 if ((old_ctx && 1801 old_ctx->replace_state == IEEE80211_CHANCTX_WILL_BE_REPLACED) || 1802 new_ctx->replace_state == IEEE80211_CHANCTX_REPLACES_OTHER) { 1803 err = ieee80211_vif_use_reserved_switch(local); 1804 if (err && err != -EAGAIN) { 1805 if (new_ctx->replace_state == 1806 IEEE80211_CHANCTX_REPLACES_OTHER) 1807 return err; 1808 1809 wiphy_info(local->hw.wiphy, 1810 "depending in-place reservation failed (err=%d)\n", 1811 err); 1812 } 1813 } 1814 1815 return 0; 1816 } 1817 1818 /* 1819 * This is similar to ieee80211_chanctx_compatible(), but rechecks 1820 * against all the links actually using it (except the one that's 1821 * passed, since that one is changing). 1822 * This is done in order to allow changes to the AP's bandwidth for 1823 * wider bandwidth OFDMA purposes, which wouldn't be treated as 1824 * compatible by ieee80211_chanctx_recheck() but is OK if the link 1825 * requesting the update is the only one using it. 1826 */ 1827 static const struct ieee80211_chan_req * 1828 ieee80211_chanctx_recheck(struct ieee80211_local *local, 1829 struct ieee80211_link_data *skip_link, 1830 struct ieee80211_chanctx *ctx, 1831 const struct ieee80211_chan_req *req, 1832 struct ieee80211_chan_req *tmp) 1833 { 1834 const struct ieee80211_chan_req *ret = req; 1835 struct ieee80211_link_data *link; 1836 1837 lockdep_assert_wiphy(local->hw.wiphy); 1838 1839 for_each_sdata_link(local, link) { 1840 if (link == skip_link) 1841 continue; 1842 1843 if (rcu_access_pointer(link->conf->chanctx_conf) == &ctx->conf) { 1844 ret = ieee80211_chanreq_compatible(ret, 1845 &link->conf->chanreq, 1846 tmp); 1847 if (!ret) 1848 return NULL; 1849 } 1850 1851 if (link->reserved_chanctx == ctx) { 1852 ret = ieee80211_chanreq_compatible(ret, 1853 &link->reserved, 1854 tmp); 1855 if (!ret) 1856 return NULL; 1857 } 1858 } 1859 1860 *tmp = *ret; 1861 return tmp; 1862 } 1863 1864 int ieee80211_link_change_chanreq(struct ieee80211_link_data *link, 1865 const struct ieee80211_chan_req *chanreq, 1866 u64 *changed) 1867 { 1868 struct ieee80211_sub_if_data *sdata = link->sdata; 1869 struct ieee80211_bss_conf *link_conf = link->conf; 1870 struct ieee80211_local *local = sdata->local; 1871 struct ieee80211_chanctx_conf *conf; 1872 struct ieee80211_chanctx *ctx; 1873 const struct ieee80211_chan_req *compat; 1874 struct ieee80211_chan_req tmp; 1875 1876 lockdep_assert_wiphy(local->hw.wiphy); 1877 1878 if (!cfg80211_chandef_usable(sdata->local->hw.wiphy, 1879 &chanreq->oper, 1880 IEEE80211_CHAN_DISABLED)) 1881 return -EINVAL; 1882 1883 /* for non-HT 20 MHz the rest doesn't matter */ 1884 if (chanreq->oper.width == NL80211_CHAN_WIDTH_20_NOHT && 1885 cfg80211_chandef_identical(&chanreq->oper, &link_conf->chanreq.oper)) 1886 return 0; 1887 1888 /* but you cannot switch to/from it */ 1889 if (chanreq->oper.width == NL80211_CHAN_WIDTH_20_NOHT || 1890 link_conf->chanreq.oper.width == NL80211_CHAN_WIDTH_20_NOHT) 1891 return -EINVAL; 1892 1893 conf = rcu_dereference_protected(link_conf->chanctx_conf, 1894 lockdep_is_held(&local->hw.wiphy->mtx)); 1895 if (!conf) 1896 return -EINVAL; 1897 1898 ctx = container_of(conf, struct ieee80211_chanctx, conf); 1899 1900 compat = ieee80211_chanctx_recheck(local, link, ctx, chanreq, &tmp); 1901 if (!compat) 1902 return -EINVAL; 1903 1904 switch (ctx->replace_state) { 1905 case IEEE80211_CHANCTX_REPLACE_NONE: 1906 if (!ieee80211_chanctx_reserved_chanreq(local, ctx, compat, 1907 &tmp)) 1908 return -EBUSY; 1909 break; 1910 case IEEE80211_CHANCTX_WILL_BE_REPLACED: 1911 /* TODO: Perhaps the bandwidth change could be treated as a 1912 * reservation itself? */ 1913 return -EBUSY; 1914 case IEEE80211_CHANCTX_REPLACES_OTHER: 1915 /* channel context that is going to replace another channel 1916 * context doesn't really exist and shouldn't be assigned 1917 * anywhere yet */ 1918 WARN_ON(1); 1919 break; 1920 } 1921 1922 ieee80211_link_update_chanreq(link, chanreq); 1923 1924 ieee80211_recalc_chanctx_chantype(local, ctx); 1925 1926 *changed |= BSS_CHANGED_BANDWIDTH; 1927 return 0; 1928 } 1929 1930 void ieee80211_link_release_channel(struct ieee80211_link_data *link) 1931 { 1932 struct ieee80211_sub_if_data *sdata = link->sdata; 1933 1934 lockdep_assert_wiphy(sdata->local->hw.wiphy); 1935 1936 if (rcu_access_pointer(link->conf->chanctx_conf)) 1937 __ieee80211_link_release_channel(link, false); 1938 } 1939 1940 void ieee80211_link_vlan_copy_chanctx(struct ieee80211_link_data *link) 1941 { 1942 struct ieee80211_sub_if_data *sdata = link->sdata; 1943 unsigned int link_id = link->link_id; 1944 struct ieee80211_bss_conf *link_conf = link->conf; 1945 struct ieee80211_bss_conf *ap_conf; 1946 struct ieee80211_local *local = sdata->local; 1947 struct ieee80211_sub_if_data *ap; 1948 struct ieee80211_chanctx_conf *conf; 1949 1950 lockdep_assert_wiphy(local->hw.wiphy); 1951 1952 if (WARN_ON(sdata->vif.type != NL80211_IFTYPE_AP_VLAN || !sdata->bss)) 1953 return; 1954 1955 ap = container_of(sdata->bss, struct ieee80211_sub_if_data, u.ap); 1956 1957 ap_conf = wiphy_dereference(local->hw.wiphy, 1958 ap->vif.link_conf[link_id]); 1959 conf = wiphy_dereference(local->hw.wiphy, 1960 ap_conf->chanctx_conf); 1961 rcu_assign_pointer(link_conf->chanctx_conf, conf); 1962 } 1963 1964 void ieee80211_iter_chan_contexts_atomic( 1965 struct ieee80211_hw *hw, 1966 void (*iter)(struct ieee80211_hw *hw, 1967 struct ieee80211_chanctx_conf *chanctx_conf, 1968 void *data), 1969 void *iter_data) 1970 { 1971 struct ieee80211_local *local = hw_to_local(hw); 1972 struct ieee80211_chanctx *ctx; 1973 1974 rcu_read_lock(); 1975 list_for_each_entry_rcu(ctx, &local->chanctx_list, list) 1976 if (ctx->driver_present) 1977 iter(hw, &ctx->conf, iter_data); 1978 rcu_read_unlock(); 1979 } 1980 EXPORT_SYMBOL_GPL(ieee80211_iter_chan_contexts_atomic); 1981