forked from mirrors/linux
		
	cfg80211: Properly track transmitting and non-transmitting BSS
When holding data of the non-transmitting BSS, we need to keep the transmitting BSS data on. Otherwise it will be released, and release the non-transmitting BSS with it. Signed-off-by: Sara Sharon <sara.sharon@intel.com> Signed-off-by: Johannes Berg <johannes.berg@intel.com>
This commit is contained in:
		
							parent
							
								
									1c8745f3ec
								
							
						
					
					
						commit
						a3584f56de
					
				
					 2 changed files with 46 additions and 2 deletions
				
			
		| 
						 | 
					@ -153,6 +153,7 @@ struct cfg80211_internal_bss {
 | 
				
			||||||
	struct list_head list;
 | 
						struct list_head list;
 | 
				
			||||||
	struct list_head hidden_list;
 | 
						struct list_head hidden_list;
 | 
				
			||||||
	struct list_head nontrans_list;
 | 
						struct list_head nontrans_list;
 | 
				
			||||||
 | 
						struct cfg80211_bss *transmitted_bss;
 | 
				
			||||||
	struct rb_node rbn;
 | 
						struct rb_node rbn;
 | 
				
			||||||
	u64 ts_boottime;
 | 
						u64 ts_boottime;
 | 
				
			||||||
	unsigned long ts;
 | 
						unsigned long ts;
 | 
				
			||||||
| 
						 | 
					@ -183,12 +184,23 @@ static inline struct cfg80211_internal_bss *bss_from_pub(struct cfg80211_bss *pu
 | 
				
			||||||
static inline void cfg80211_hold_bss(struct cfg80211_internal_bss *bss)
 | 
					static inline void cfg80211_hold_bss(struct cfg80211_internal_bss *bss)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	atomic_inc(&bss->hold);
 | 
						atomic_inc(&bss->hold);
 | 
				
			||||||
 | 
						if (bss->transmitted_bss) {
 | 
				
			||||||
 | 
							bss = container_of(bss->transmitted_bss,
 | 
				
			||||||
 | 
									   struct cfg80211_internal_bss, pub);
 | 
				
			||||||
 | 
							atomic_inc(&bss->hold);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static inline void cfg80211_unhold_bss(struct cfg80211_internal_bss *bss)
 | 
					static inline void cfg80211_unhold_bss(struct cfg80211_internal_bss *bss)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int r = atomic_dec_return(&bss->hold);
 | 
						int r = atomic_dec_return(&bss->hold);
 | 
				
			||||||
	WARN_ON(r < 0);
 | 
						WARN_ON(r < 0);
 | 
				
			||||||
 | 
						if (bss->transmitted_bss) {
 | 
				
			||||||
 | 
							bss = container_of(bss->transmitted_bss,
 | 
				
			||||||
 | 
									   struct cfg80211_internal_bss, pub);
 | 
				
			||||||
 | 
							r = atomic_dec_return(&bss->hold);
 | 
				
			||||||
 | 
							WARN_ON(r < 0);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -110,6 +110,12 @@ static inline void bss_ref_get(struct cfg80211_registered_device *rdev,
 | 
				
			||||||
				   pub);
 | 
									   pub);
 | 
				
			||||||
		bss->refcount++;
 | 
							bss->refcount++;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if (bss->transmitted_bss) {
 | 
				
			||||||
 | 
							bss = container_of(bss->transmitted_bss,
 | 
				
			||||||
 | 
									   struct cfg80211_internal_bss,
 | 
				
			||||||
 | 
									   pub);
 | 
				
			||||||
 | 
							bss->refcount++;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static inline void bss_ref_put(struct cfg80211_registered_device *rdev,
 | 
					static inline void bss_ref_put(struct cfg80211_registered_device *rdev,
 | 
				
			||||||
| 
						 | 
					@ -126,6 +132,18 @@ static inline void bss_ref_put(struct cfg80211_registered_device *rdev,
 | 
				
			||||||
		if (hbss->refcount == 0)
 | 
							if (hbss->refcount == 0)
 | 
				
			||||||
			bss_free(hbss);
 | 
								bss_free(hbss);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (bss->transmitted_bss) {
 | 
				
			||||||
 | 
							struct cfg80211_internal_bss *tbss;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							tbss = container_of(bss->transmitted_bss,
 | 
				
			||||||
 | 
									    struct cfg80211_internal_bss,
 | 
				
			||||||
 | 
									    pub);
 | 
				
			||||||
 | 
							tbss->refcount--;
 | 
				
			||||||
 | 
							if (tbss->refcount == 0)
 | 
				
			||||||
 | 
								bss_free(tbss);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	bss->refcount--;
 | 
						bss->refcount--;
 | 
				
			||||||
	if (bss->refcount == 0)
 | 
						if (bss->refcount == 0)
 | 
				
			||||||
		bss_free(bss);
 | 
							bss_free(bss);
 | 
				
			||||||
| 
						 | 
					@ -1024,6 +1042,7 @@ static bool cfg80211_combine_bsses(struct cfg80211_registered_device *rdev,
 | 
				
			||||||
static struct cfg80211_internal_bss *
 | 
					static struct cfg80211_internal_bss *
 | 
				
			||||||
cfg80211_bss_update(struct cfg80211_registered_device *rdev,
 | 
					cfg80211_bss_update(struct cfg80211_registered_device *rdev,
 | 
				
			||||||
		    struct cfg80211_internal_bss *tmp,
 | 
							    struct cfg80211_internal_bss *tmp,
 | 
				
			||||||
 | 
							    struct cfg80211_bss *trans_bss,
 | 
				
			||||||
		    bool signal_valid)
 | 
							    bool signal_valid)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct cfg80211_internal_bss *found = NULL;
 | 
						struct cfg80211_internal_bss *found = NULL;
 | 
				
			||||||
| 
						 | 
					@ -1181,6 +1200,17 @@ cfg80211_bss_update(struct cfg80211_registered_device *rdev,
 | 
				
			||||||
			goto drop;
 | 
								goto drop;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/* This must be before the call to bss_ref_get */
 | 
				
			||||||
 | 
							if (trans_bss) {
 | 
				
			||||||
 | 
								struct cfg80211_internal_bss *pbss =
 | 
				
			||||||
 | 
									container_of(trans_bss,
 | 
				
			||||||
 | 
										     struct cfg80211_internal_bss,
 | 
				
			||||||
 | 
										     pub);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								new->transmitted_bss = trans_bss;
 | 
				
			||||||
 | 
								bss_ref_get(rdev, pbss);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		list_add_tail(&new->list, &rdev->bss_list);
 | 
							list_add_tail(&new->list, &rdev->bss_list);
 | 
				
			||||||
		rdev->bss_entries++;
 | 
							rdev->bss_entries++;
 | 
				
			||||||
		rb_insert_bss(rdev, new);
 | 
							rb_insert_bss(rdev, new);
 | 
				
			||||||
| 
						 | 
					@ -1336,7 +1366,8 @@ cfg80211_inform_single_bss_data(struct wiphy *wiphy,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	signal_valid = abs(data->chan->center_freq - channel->center_freq) <=
 | 
						signal_valid = abs(data->chan->center_freq - channel->center_freq) <=
 | 
				
			||||||
		wiphy->max_adj_channel_rssi_comp;
 | 
							wiphy->max_adj_channel_rssi_comp;
 | 
				
			||||||
	res = cfg80211_bss_update(wiphy_to_rdev(wiphy), &tmp, signal_valid);
 | 
						res = cfg80211_bss_update(wiphy_to_rdev(wiphy), &tmp, trans_bss,
 | 
				
			||||||
 | 
									  signal_valid);
 | 
				
			||||||
	if (!res)
 | 
						if (!res)
 | 
				
			||||||
		return NULL;
 | 
							return NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1639,7 +1670,8 @@ cfg80211_inform_single_bss_frame_data(struct wiphy *wiphy,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	signal_valid = abs(data->chan->center_freq - channel->center_freq) <=
 | 
						signal_valid = abs(data->chan->center_freq - channel->center_freq) <=
 | 
				
			||||||
		wiphy->max_adj_channel_rssi_comp;
 | 
							wiphy->max_adj_channel_rssi_comp;
 | 
				
			||||||
	res = cfg80211_bss_update(wiphy_to_rdev(wiphy), &tmp, signal_valid);
 | 
						res = cfg80211_bss_update(wiphy_to_rdev(wiphy), &tmp, trans_bss,
 | 
				
			||||||
 | 
									  signal_valid);
 | 
				
			||||||
	if (!res)
 | 
						if (!res)
 | 
				
			||||||
		return NULL;
 | 
							return NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue