// // Copyright (C) 2014 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #include "apmanager/device.h" #include <base/strings/stringprintf.h> #include <brillo/strings/string_utils.h> #include <shill/net/attribute_list.h> #include <shill/net/ieee80211.h> #include "apmanager/config.h" #include "apmanager/control_interface.h" #include "apmanager/manager.h" using shill::ByteString; using std::string; namespace apmanager { Device::Device(Manager* manager, const string& device_name, int identifier) : manager_(manager), supports_ap_mode_(false), identifier_(identifier), adaptor_(manager->control_interface()->CreateDeviceAdaptor(this)) { SetDeviceName(device_name); SetInUse(false); } Device::~Device() {} void Device::RegisterInterface(const WiFiInterface& new_interface) { LOG(INFO) << "RegisteringInterface " << new_interface.iface_name << " on device " << GetDeviceName(); for (const auto& interface : interface_list_) { // Done if interface already in the list. if (interface.iface_index == new_interface.iface_index) { LOG(INFO) << "Interface " << new_interface.iface_name << " already registered."; return; } } interface_list_.push_back(new_interface); UpdatePreferredAPInterface(); } void Device::DeregisterInterface(const WiFiInterface& interface) { LOG(INFO) << "DeregisteringInterface " << interface.iface_name << " on device " << GetDeviceName(); for (auto it = interface_list_.begin(); it != interface_list_.end(); ++it) { if (it->iface_index == interface.iface_index) { interface_list_.erase(it); UpdatePreferredAPInterface(); return; } } } void Device::ParseWiphyCapability(const shill::Nl80211Message& msg) { // Parse NL80211_ATTR_SUPPORTED_IFTYPES for AP mode interface support. shill::AttributeListConstRefPtr supported_iftypes; if (!msg.const_attributes()->ConstGetNestedAttributeList( NL80211_ATTR_SUPPORTED_IFTYPES, &supported_iftypes)) { LOG(ERROR) << "NL80211_CMD_NEW_WIPHY had no NL80211_ATTR_SUPPORTED_IFTYPES"; return; } supported_iftypes->GetFlagAttributeValue(NL80211_IFTYPE_AP, &supports_ap_mode_); // Parse WiFi band capabilities. shill::AttributeListConstRefPtr wiphy_bands; if (!msg.const_attributes()->ConstGetNestedAttributeList( NL80211_ATTR_WIPHY_BANDS, &wiphy_bands)) { LOG(ERROR) << "NL80211_CMD_NEW_WIPHY had no NL80211_ATTR_WIPHY_BANDS"; return; } shill::AttributeIdIterator band_iter(*wiphy_bands); for (; !band_iter.AtEnd(); band_iter.Advance()) { BandCapability band_cap; shill::AttributeListConstRefPtr wiphy_band; if (!wiphy_bands->ConstGetNestedAttributeList(band_iter.GetId(), &wiphy_band)) { LOG(WARNING) << "WiFi band " << band_iter.GetId() << " not found"; continue; } // ...Each band has a FREQS attribute... shill::AttributeListConstRefPtr frequencies; if (!wiphy_band->ConstGetNestedAttributeList(NL80211_BAND_ATTR_FREQS, &frequencies)) { LOG(ERROR) << "BAND " << band_iter.GetId() << " had no 'frequencies' attribute"; continue; } // ...And each FREQS attribute contains an array of information about the // frequency... shill::AttributeIdIterator freq_iter(*frequencies); for (; !freq_iter.AtEnd(); freq_iter.Advance()) { shill::AttributeListConstRefPtr frequency; if (frequencies->ConstGetNestedAttributeList(freq_iter.GetId(), &frequency)) { // ...Including the frequency, itself (the part we want). uint32_t frequency_value = 0; if (frequency->GetU32AttributeValue(NL80211_FREQUENCY_ATTR_FREQ, &frequency_value)) { band_cap.frequencies.push_back(frequency_value); } } } wiphy_band->GetU16AttributeValue(NL80211_BAND_ATTR_HT_CAPA, &band_cap.ht_capability_mask); wiphy_band->GetU16AttributeValue(NL80211_BAND_ATTR_VHT_CAPA, &band_cap.vht_capability_mask); band_capability_.push_back(band_cap); } } bool Device::ClaimDevice(bool full_control) { if (GetInUse()) { LOG(ERROR) << "Failed to claim device [" << GetDeviceName() << "]: already in used."; return false; } if (full_control) { for (const auto& interface : interface_list_) { manager_->ClaimInterface(interface.iface_name); claimed_interfaces_.insert(interface.iface_name); } } else { manager_->ClaimInterface(GetPreferredApInterface()); claimed_interfaces_.insert(GetPreferredApInterface()); } SetInUse(true); return true; } bool Device::ReleaseDevice() { if (!GetInUse()) { LOG(ERROR) << "Failed to release device [" << GetDeviceName() << "]: not currently in-used."; return false; } for (const auto& interface : claimed_interfaces_) { manager_->ReleaseInterface(interface); } claimed_interfaces_.clear(); SetInUse(false); return true; } bool Device::InterfaceExists(const string& interface_name) { for (const auto& interface : interface_list_) { if (interface.iface_name == interface_name) { return true; } } return false; } bool Device::GetHTCapability(uint16_t channel, string* ht_cap) { // Get the band capability based on the channel. BandCapability band_cap; if (!GetBandCapability(channel, &band_cap)) { LOG(ERROR) << "No band capability found for channel " << channel; return false; } std::vector<string> ht_capability; // LDPC coding capability. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskLdpcCoding) { ht_capability.push_back("LDPC"); } // Supported channel width set. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskSupWidth2040) { // Determine secondary channel is below or above the primary. bool above = false; if (!GetHTSecondaryChannelLocation(channel, &above)) { LOG(ERROR) << "Unable to determine secondary channel location for " << "channel " << channel; return false; } if (above) { ht_capability.push_back("HT40+"); } else { ht_capability.push_back("HT40-"); } } // Spatial Multiplexing (SM) Power Save. uint16_t power_save_mask = (band_cap.ht_capability_mask >> shill::IEEE_80211::kHTCapMaskSmPsShift) & 0x3; if (power_save_mask == 0) { ht_capability.push_back("SMPS-STATIC"); } else if (power_save_mask == 1) { ht_capability.push_back("SMPS-DYNAMIC"); } // HT-greenfield. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskGrnFld) { ht_capability.push_back("GF"); } // Short GI for 20 MHz. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskSgi20) { ht_capability.push_back("SHORT-GI-20"); } // Short GI for 40 MHz. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskSgi40) { ht_capability.push_back("SHORT-GI-40"); } // Tx STBC. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskTxStbc) { ht_capability.push_back("TX-STBC"); } // Rx STBC. uint16_t rx_stbc = (band_cap.ht_capability_mask >> shill::IEEE_80211::kHTCapMaskRxStbcShift) & 0x3; if (rx_stbc == 1) { ht_capability.push_back("RX-STBC1"); } else if (rx_stbc == 2) { ht_capability.push_back("RX-STBC12"); } else if (rx_stbc == 3) { ht_capability.push_back("RX-STBC123"); } // HT-delayed Block Ack. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskDelayBA) { ht_capability.push_back("DELAYED-BA"); } // Maximum A-MSDU length. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskMaxAmsdu) { ht_capability.push_back("MAX-AMSDU-7935"); } // DSSS/CCK Mode in 40 MHz. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMaskDsssCck40) { ht_capability.push_back("DSSS_CCK-40"); } // 40 MHz intolerant. if (band_cap.ht_capability_mask & shill::IEEE_80211::kHTCapMask40MHzIntolerant) { ht_capability.push_back("40-INTOLERANT"); } *ht_cap = base::StringPrintf("[%s]", brillo::string_utils::Join(" ", ht_capability).c_str()); return true; } bool Device::GetVHTCapability(uint16_t channel, string* vht_cap) { // TODO(zqiu): to be implemented. return false; } void Device::SetDeviceName(const std::string& device_name) { adaptor_->SetDeviceName(device_name); } string Device::GetDeviceName() const { return adaptor_->GetDeviceName(); } void Device::SetPreferredApInterface(const std::string& interface_name) { adaptor_->SetPreferredApInterface(interface_name); } string Device::GetPreferredApInterface() const { return adaptor_->GetPreferredApInterface(); } void Device::SetInUse(bool in_use) { return adaptor_->SetInUse(in_use); } bool Device::GetInUse() const { return adaptor_->GetInUse(); } // static bool Device::GetHTSecondaryChannelLocation(uint16_t channel, bool* above) { bool ret_val = true; // Determine secondary channel location base on the channel. Refer to // ht_cap section in hostapd.conf documentation. switch (channel) { case 7: case 8: case 9: case 10: case 11: case 12: case 13: case 40: case 48: case 56: case 64: *above = false; break; case 1: case 2: case 3: case 4: case 5: case 6: case 36: case 44: case 52: case 60: *above = true; break; default: ret_val = false; break; } return ret_val; } bool Device::GetBandCapability(uint16_t channel, BandCapability* capability) { uint32_t frequency; if (!Config::GetFrequencyFromChannel(channel, &frequency)) { LOG(ERROR) << "Invalid channel " << channel; return false; } for (const auto& band : band_capability_) { if (std::find(band.frequencies.begin(), band.frequencies.end(), frequency) != band.frequencies.end()) { *capability = band; return true; } } return false; } void Device::UpdatePreferredAPInterface() { // Return if device doesn't support AP interface mode. if (!supports_ap_mode_) { return; } // Use the first registered AP mode interface if there is one, otherwise use // the first registered managed mode interface. If none are available, then // no interface can be used for AP operation on this device. WiFiInterface preferred_interface; for (const auto& interface : interface_list_) { if (interface.iface_type == NL80211_IFTYPE_AP) { preferred_interface = interface; break; } else if (interface.iface_type == NL80211_IFTYPE_STATION && preferred_interface.iface_name.empty()) { preferred_interface = interface; } // Ignore all other interface types. } // Update preferred AP interface property. SetPreferredApInterface(preferred_interface.iface_name); } } // namespace apmanager