Localização de Wi-Fi: alcance com RTT

Você pode usar o recurso de localização de Wi-Fi fornecido pela API Wi-Fi RTT (Round-Trip-Time) para medir a distância dos pontos de acesso e dos dispositivos Wi-Fi Aware próximos e compatíveis com RTT.

Ao medir a distância de até três ou mais pontos de acesso, será possível usar um algoritmo de multilateração para estimar a posição do dispositivo que melhor se encaixa nessas medidas. Normalmente, o resultado é preciso a uma distância de 1 a 2 metros.

Com esse nível de precisão, é possível criar serviços aprimorados baseados na localização, como a navegação interna, o controle de voz sem ambiguidades (por exemplo, "acenda esta luz") e informações baseadas na localização (por exemplo, "este produto tem ofertas especiais?").

O dispositivo que faz a solicitação não precisa se conectar aos pontos de acesso para medir a distância com o Wi-Fi RTT. Para manter a privacidade, somente o dispositivo solicitante pode para determinar a distância até o ponto de acesso; os pontos de acesso não têm essas informações. As operações de Wi-Fi RTT são ilimitadas para aplicativos em primeiro plano, mas não para os em segundo plano.

O RTT do Wi-Fi e os recursos de Medição de tempo fino (FTM, na sigla em inglês) relacionados são especificados pela norma IEEE 802.11-2016. O Wi-Fi RTT exige o uso da medição de tempo precisa fornecida pelo FTM, já que ele calcula a distância medindo o tempo que um pacote leva para fazer uma viagem de ida e volta entre dois dispositivos e depois multiplica esse tempo pela velocidade da luz.

O Android 15 (nível 35 da API) introduziu suporte a IEEE 802.11az sem gatilho (NTB).

Diferenças de implementação baseadas na versão Android

O Wi-Fi RTT foi introduzido no Android 9 (API de nível 28). Para usar esse protocolo para determinar a posição de um dispositivo usando multilateration com dispositivos que executam o Android 9, você precisará ter acesso aos dados de localização do ponto de acesso (AP, na sigla em inglês) predeterminados no seu aplicativo. Você pode decidir a melhor forma de armazenar e recuperar esses dados.

Em dispositivos que executam Android 10 (API de nível 29) e versões mais recentes, os dados da localização do ponto de acesso podem ser representados por objetos ResponderLocation, que incluem latitude, longitude e altitude. Para os pontos de acesso do Wi-Fi RTT compatíveis com informações de configuração da localização/relatório cívico de localização (dados de LCI/LCR), o protocolo retornará um objeto ResponderLocation durante o processo de alcance.

Esse recurso permite que os aplicativos consultem diretamente os pontos de acesso para solicitar a posição deles, em vez de precisar armazenar essas informações antecipadamente. Dessa forma, seu aplicativo poderá encontrar pontos de acesso e determinar as posições relacionadas, mesmo que esses pontos sejam desconhecidos. Por exemplo, quando um usuário entra em um novo prédio.

O suporte ao alcance NTB IEEE 802.11az está disponível em dispositivos com o Android 15 (nível 35 da API) e mais recentes. Isso significa que, se o dispositivo oferecer suporte a IEEE 802.11az, Modo de resposta do NTB (indicado por WifiRttManager.CHARACTERISTICS_KEY_BOOLEAN_STA_RESPONDER), seu app poderá encontrar APs compatíveis com IEEE 802.11mc e IEEE 802.11az com um único de intervalo de endereços IP. A API RangingResult foi estendida para fornecer informações sobre os valores mínimo e máximo que podem ser usados para o intervalo entre o alcance de medições, deixando o intervalo exato no controle do aplicativo.

Requisitos

  • O hardware do dispositivo que faz a solicitação de alcance deve implementar a Padrão FTM 802.11-2016 ou padrão 802.11az (variação não baseada em gatilhos).
  • O dispositivo que faz a solicitação de alcance precisa executar o Android 9 (API de nível 28) ou versão mais recente. O alcance não baseado em acionador do IEEE 802.11az está ativado nos dispositivos com o Android 15 (nível 35 da API) e versões mais recentes.
  • O dispositivo que faz a solicitação de alcance precisa ter a busca por Wi-Fi e os serviços de localização ativados (em Configurações > Localização).
  • Se o app que faz a solicitação de alcance segmentar Android 13 (nível 33 da API) ou versões mais recentes, precisa ter a NEARBY_WIFI_DEVICES permissão. Se um aplicativo é direcionado a uma versão anterior do Android, ele precisa têm os ACCESS_FINE_LOCATION no lugar da permissão.
  • O aplicativo deve consultar o alcance de pontos de acesso enquanto o aplicativo está visível ou em um serviço em primeiro plano. O aplicativo não pode acessar as informações de localização em segundo plano.
  • O ponto de acesso precisa implementar o padrão FTM IEEE 802.11-2016 ou IEEE Padrão 802.11az (intervalo não baseado em gatilho).

Configurar

Se você quiser configurar o aplicativo para usar o Wi-Fi RTT, siga estas etapas:

1. Solicite permissões

Solicite as seguintes permissões no manifesto do app:

<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<!-- If your app targets Android 13 (API level 33)
     or higher, you must declare the NEARBY_WIFI_DEVICES permission. -->
<uses-permission android:name="android.permission.NEARBY_WIFI_DEVICES"
                 <!-- If your app derives location information from Wi-Fi APIs,
                      don't include the "usesPermissionFlags" attribute. -->
                 android:usesPermissionFlags="neverForLocation" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"
                 <!-- If any feature in your app relies on precise location
                      information, don't include the "maxSdkVersion"
                      attribute. -->
                 android:maxSdkVersion="32" />

As permissões NEARBY_WIFI_DEVICES e ACCESS_FINE_LOCATION são perigosas portanto, você precisa solicitá-las no momento da execução sempre que o usuário quiser executar uma operação de verificação de RTT. Seu app precisará solicitar a permissão se ela ainda não tiver sido concedida. Para mais informações sobre permissões de execução, consulte Solicite permissões do app.

2. Verifique se o dispositivo é compatível com Wi-Fi RTT

Para verificar se o dispositivo é compatível com Wi-Fi RTT, use o API PackageManager:

Kotlin

context.packageManager.hasSystemFeature(PackageManager.FEATURE_WIFI_RTT)

Java

context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_WIFI_RTT);

3. Verifique se o Wi-Fi RTT está disponível

O Wi-Fi RTT pode estar presente no dispositivo, mas não estar disponível porque o usuário desativou o Wi-Fi. Dependendo dos recursos de hardware e firmware, alguns Os dispositivos podem não oferecer suporte ao Wi-Fi RTT se o SoftAP ou o tethering estiverem em uso. Para verificar se o Wi-Fi RTT está disponível, ligue isAvailable()

A disponibilidade do Wi-Fi RTT pode mudar a qualquer momento. Seu aplicativo deve registrar um BroadcastReceiver para receber ACTION_WIFI_RTT_STATE_CHANGED, que é enviado quando a disponibilidade muda. Ao receber o intent de transmissão, seu aplicativo deverá verificar o estado atual da disponibilidade e ajustar o próprio comportamento de acordo com essa informação.

Por exemplo:

Kotlin

val filter = IntentFilter(WifiRttManager.ACTION_WIFI_RTT_STATE_CHANGED)
val myReceiver = object: BroadcastReceiver() {

    override fun onReceive(context: Context, intent: Intent) {
        if (wifiRttManager.isAvailable) {
            …
        } else {
            …
        }
    }
}
context.registerReceiver(myReceiver, filter)

Java

IntentFilter filter =
    new IntentFilter(WifiRttManager.ACTION_WIFI_RTT_STATE_CHANGED);
BroadcastReceiver myReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        if (wifiRttManager.isAvailable()) {
            …
        } else {
            …
        }
    }
};
context.registerReceiver(myReceiver, filter);

Para mais informações, consulte Visão geral de transmissões.

Criar uma solicitação de alcance

Uma solicitação de alcance (RangingRequest) é criado especificando uma lista de APs ou pontos Wi-Fi Aware aos quais um intervalo é solicitado. É possível especificar vários pontos de acesso ou pontos do Wi-Fi Aware em uma única solicitação. As distâncias para todos os dispositivos são medidas e retornadas.

Por exemplo, uma solicitação pode usar addAccessPoint() para especificar um ponto de acesso para medir a distância:

Kotlin

val req: RangingRequest = RangingRequest.Builder().run {
    addAccessPoint(ap1ScanResult)
    addAccessPoint(ap2ScanResult)
    build()
}

Java

RangingRequest.Builder builder = new RangingRequest.Builder();
builder.addAccessPoint(ap1ScanResult);
builder.addAccessPoint(ap2ScanResult);

RangingRequest req = builder.build();

Um ponto de acesso é identificado pela objeto ScanResult, que pode ser conseguido chamando WifiManager.getScanResults() Você pode usar addAccessPoints(List<ScanResult>) para adicionar vários pontos de acesso em um lote.

Objetos ScanResult podem conter IEEE 802.11mc (is80211mcResponder()) e Compatível com alcance não baseado em gatilho (is80211azNtbResponder()) IEEE 802.11az APs. Os dispositivos compatíveis com o intervalo NTB IEEE 802.11az executam 802.11mc ou 802.11az variando dependendo da capacidade do AP, deixando o padrão 802.11az quando o AP oferece suporte a ambos. Dispositivos que não têm suporte a IEEE 802.11az realizam todas usando o protocolo IEEE 802.11mc.

Da mesma forma, uma solicitação de alcance pode adicionar um ponto Wi-Fi Aware usando o MAC ou o PeerHandle dele, usando as addWifiAwarePeer(MacAddress peer) e addWifiAwarePeer(PeerHandle peer) , respectivamente. Para mais informações sobre como descobrir pontos do Wi-Fi Aware, consulte a documentação do Wi-Fi Aware.

Solicitar o alcance

Um app emite uma solicitação de alcance usando o WifiRttManager.startRanging() e forneça o seguinte: RangingRequest para especificar operação, um Executor para especificar o contexto de callback e uma RangingResultCallback para receber os resultados.

Exemplo:

Kotlin

val mgr = context.getSystemService(Context.WIFI_RTT_RANGING_SERVICE) as WifiRttManager
val request: RangingRequest = myRequest
mgr.startRanging(request, executor, object : RangingResultCallback() {

    override fun onRangingResults(results: List<RangingResult>) { … }

    override fun onRangingFailure(code: Int) { … }
})

Java

WifiRttManager mgr =
      (WifiRttManager) Context.getSystemService(Context.WIFI_RTT_RANGING_SERVICE);

RangingRequest request ...;
mgr.startRanging(request, executor, new RangingResultCallback() {

  @Override
  public void onRangingFailure(int code) { … }

  @Override
  public void onRangingResults(List<RangingResult> results) { … }
});

A operação de alcance é executada de forma assíncrona, e os resultados de alcance são retornado em um dos callbacks RangingResultCallback:

  • Se toda a operação de alcance falhar, o onRangingFailure callback é acionado com um código de status descrito em RangingResultCallback Esse erro ocorrerá caso o serviço não possa executar uma operação de alcance no momento. Por exemplo, o Wi-Fi está desativado, o aplicativo ultrapassou o limite de solicitações de alcance ou devido a um problema de permissão.
  • Quando a operação de alcance for concluída, o onRangingResults callback é acionado com uma lista de resultados que corresponde à lista de solicitações, um resultado para cada solicitação. A ordem dos resultados não corresponderá necessariamente à das solicitações. A operação de alcance talvez seja concluída, mas cada resultado ainda poderá indicar uma falha nessa medição específica.

Interpretar os resultados de alcance

Cada um dos resultados retornados pelo onRangingResults callback é especificado por um RangingResult. objeto. Em cada solicitação, faça o seguinte:

1. Identifique a solicitação

Identifique a solicitação com base nas informações fornecidas ao criar o RangingRequest: na maioria das vezes, um endereço MAC fornecido no ScanResult identificando um ponto O endereço MAC pode ser obtido a partir do resultado de alcance usando o getMacAddress() .

A lista de resultados de alcance pode estar em ordem diferente dos pontos (pontos de acesso) especificados na solicitação. Por isso, use o endereço MAC para identificar o ponto, não a ordem dos resultados.

2. Determine se cada medição foi concluída

Para determinar se uma medição foi bem-sucedida, use o getStatus() . Qualquer valor que não seja STATUS_SUCCESS indica uma falha. Uma falha significa que todos os outros campos deste resultado (exceto a identificação da solicitação acima) são inválidos, e o O método get* vai falhar com uma IllegalStateException.

3. Veja os resultados para cada medição concluída

Para cada medição bem-sucedida (RangingResult), é possível recuperar o resultado com os respectivos métodos get:

.

Dispositivos Android compatíveis com Wi-Fi RTT

As tabelas a seguir listam alguns smartphones, pontos de acesso e dispositivos de centros de distribuição, armazenamento e varejo. compatíveis com Wi-Fi-RTT. Elas não abrangem todos os dispositivos compatíveis. Recomendamos que você entre em contato conosco para listar seus produtos compatíveis com RTT aqui.

Pontos de acesso

Fabricante e Modelo Data de suporte
Nest Wifi Pro (Wi-Fi 6E) Compatível
Compulab WILD AP Compatível
Google Wi-Fi Compatível
Roteador Google Nest Wifi Compatível
Ponto Google Nest Wifi Compatível
Aruba AP-635 (link em espanhol) Compatível
Cisco 9130 Compatível
Cisco 9136 Compatível
Cisco 9166 Compatível
Cisco 9164 Compatível
AP-505 (Aruba) Compatível
AP-515 de Aruba Compatível
AP-575 (Aruba) Compatível
AP-518 de Aruba Compatível
AP-505H de Aruba Compatível
AP-565 (Aruba) Compatível
AP-535 de Aruba Compatível

Smartphones

Fabricante e Modelo Versão do Android
Pixel 6 9.0+
Pixel 6 Pro 9.0+
Pixel 5 9.0+
Pixel 5a 9.0+
Pixel 5a (5G) 9.0+
Xiaomi Mi 10 Pro 9.0+
Xiaomi Mi 10 9.0+
Xiaomi RedMi 9T Pro 9.0+
Xiaomi Mi 9T 9.0+
Xiaomi Mi 9 9.0+
Xiaomi Mi Note 10 9.0+
Xiaomi Mi Note 10 Lite 9.0+
Xiaomi RedMi Note 9S 9.0+
Xiaomi RedMi Note 9 Pro 9.0+
Xiaomi RedMi Note 8T 9.0+
Xiaomi RedMi Note 8 9.0+
Xiaomi RedMi K30 Pro 9.0+
Xiaomi RedMi K20 Pro 9.0+
Xiaomi RedMi K20 9.0+
Xiaomi Redmi Note 5 Pro 9.0+
Xiaomi Mi CC9 Pro 9.0+
LG G8X ThinQ 9.0+
LG V50S ThinQ 9.0+
LG V60 ThinQ 9.0+
LG V30 9.0+
Samsung Galaxy Note 10+ 5G 9.0+
Samsung Galaxy S20+ 5G 9.0+
Samsung Galaxy S20+ 9.0+
Samsung Galaxy S20 5G 9.0+
Samsung Galaxy S20 Ultra 5G 9.0+
Samsung Galaxy S20 9.0+
Samsung Galaxy Note 10 ou mais recente 9.0+
Samsung Galaxy Note 10 5G 9.0+
Samsung Galaxy Note 10 9.0+
Samsung A9 Pro 9.0+
Google Pixel 4 XL 9.0+
Google Pixel 4 9.0+
Google Pixel 4a 9.0+
Google Pixel 3 XL 9.0+
Google Pixel 3 9.0+
Google Pixel 3a XL 9.0+
Google Pixel 3a 9.0+
Google Pixel 2 XL 9.0+
Google Pixel 2 9.0+
Google Pixel 1 XL 9.0+
Google Pixel 1 9.0+
Poco x2 9.0+
Sharp Aquos R3 SH-04L 9.0+

Dispositivos para centros de distribuição, armazenamento e varejo

Fabricante e Modelo Versão do Android
Zebra PS20 10.0 ou superior
Zebra TC52/TC52HC 10.0 ou superior
Zebra TC57 10.0 ou superior
Zebra TC72 10.0 ou superior
Zebra TC77 10.0 ou superior
Zebra MC93 10.0 ou superior
Zebra TC8300 10.0 ou superior
Zebra VC8300 10.0 ou superior
Zebra EC30 10.0 ou superior
Zebra ET51 10.0 ou superior
Zebra ET56 10.0 ou superior
Zebra L10 10.0 ou superior
Zebra CC600/CC6000 10.0 ou superior
Zebra MC3300x 10.0 ou superior
Zebra MC330x 10.0 ou superior
Zebra TC52x 10.0 ou superior
Zebra TC57x 10.0 ou superior
Zebra EC50 (LAN e HC) 10.0 ou superior
Zebra EC55 (WAN) 10.0 ou superior
Zebra WT6300 10.0 ou superior
Skorpio X5 (link em inglês) 10.0 ou superior