Tamanho de página de 16 KB

O tamanho da página é a granularidade em que um SO gerencia a memória. A maioria das CPUs atuais oferece suporte a um tamanho de página de 4 KB. Por isso, o SO e os apps Android foram criados e otimizados para funcionar com um tamanho de página de 4 KB. As CPUs ARM oferecem suporte ao tamanho de página maior de 16 KB e, a partir do Android 15, o AOSP também oferece suporte à criação do Android com um tamanho de página de 16 KB. Essa opção usa mais memória, mas melhora o desempenho do sistema. A partir do Android 15, essa opção não é ativada por padrão, mas está disponível como um modo de desenvolvedor ou uma opção para OEMs e desenvolvedores de apps se prepararem para a mudança para o modo de 16 KB em todos os lugares no futuro.

O Android 15 e versões mais recentes oferecem suporte à criação do Android com um alinhamento ELF de 16 KB, que funciona com kernels de 4 KB e 16 KB, começando com android14-6.1. Quando usada com um kernel de 16 KB, essa configuração usa mais memória, mas melhora o desempenho do sistema.

Definir o espaço do usuário do Android como 16 KB

As páginas de 16 KB têm suporte apenas em destinos arm64 com núcleos de 16 KB. No entanto, também há uma opção para simular 16 KB de espaço do usuário em x86_64 para Cuttlefish.

Para destinos arm64, se você usar o Kleaf para criar o kernel, o --page_size=16k vai criar o kernel no modo de 16 KB. Se você estiver usando diretamente a configuração do kernel do Linux, poderá selecionar páginas de 16 KB definindo CONFIG_ARM64_16K_PAGES em vez de CONFIG_ARM64_4K_PAGES.

Para ativar o suporte ao tamanho de página de 16 KB no espaço do usuário do Android, defina as seguintes opções de build no seu produto:

  • PRODUCT_NO_BIONIC_PAGE_SIZE_MACRO := true remove a definição PAGE_SIZE e faz com que os componentes determinem o tamanho da página no momento da execução.
  • PRODUCT_MAX_PAGE_SIZE_SUPPORTED := 16384, que garante que os arquivos ELF da plataforma sejam criados com alinhamento de 16 KB. Esse tamanho maior do que o necessário é para compatibilidade futura. Com o alinhamento ELF de 16 KB, o kernel pode oferecer suporte a tamanhos de página de 4 KB/16 KB.

Verificar flags de build

Depois de selecionar o destino lunch, verifique se as flags de build estão configuradas corretamente no ambiente:

$ source build/envsetup.sh
$ lunch target

$ get_build_var TARGET_MAX_PAGE_SIZE_SUPPORTED
16384
$ get_build_var TARGET_NO_BIONIC_PAGE_SIZE_MACRO
true

Se os dois comandos anteriores retornarem 16384 e true, respectivamente, os flags de build estão configurados corretamente para funcionar com um kernel de 16 KB. No entanto, mesmo que um build seja aprovado, ainda pode haver problemas de execução devido a diferenças em um ambiente de 16 KB.

Programação do sistema de tamanho de página de 16 KB

A grande maioria do código em qualquer dispositivo Android não lida diretamente com o tamanho da página. No entanto, para códigos que lidam com páginas, o comportamento de alocação de memória do kernel muda, e você precisa levar isso em consideração para escrever um código que seja compatível e tenha o máximo de desempenho e uso mínimo de recursos.

Se você chamar mmap em uma região de 1 KB, 2 KB ou até 4 KB em um sistema de 4 KB, o sistema reserva 4 KB para implementar isso. Em outras palavras, ao solicitar memória do kernel, ele precisa sempre arredondar a memória solicitada para o tamanho de página mais próximo. Por exemplo, se você alocar uma região de 5 KB em uma região de 4 KB, o kernel alocará 8 KB.

Em um kernel de 16 KB, essas "pontas de cauda" extras das páginas são maiores. Por exemplo, todas essas alocações, de 1 KB a 5 KB, alocariam 16 KB quando usadas com um kernel de 16 KB. Se você solicitar 17 KB, ele alocará 32 KB.

Por exemplo, em um sistema de 4 KB, é possível alocar duas regiões anônimas de leitura e gravação de 4 KB. No entanto, em um kernel de 16 KB, isso resultaria na alocação de duas páginas ou 32 KB. Em um kernel de 16 KB, se possível, essas regiões podem ser combinadas em uma única página de leitura ou gravável para que apenas 16 KB sejam usados, desperdiando 8 KB em comparação com o caso de kernel de 4 KB. Para reduzir ainda mais o uso de memória, mais páginas podem ser combinadas. Na verdade, em um sistema de 16 KB otimizado ao máximo, as páginas de 16 KB exigem menos memória do que os sistemas de 4 KB porque a tabela de páginas tem um quarto do tamanho para a mesma memória.

Sempre que usar mmap, arredonde o tamanho solicitado para o tamanho de página mais próximo. Isso garante que toda a quantidade de memória alocada pelo kernel seja diretamente visível para o espaço do usuário nos valores de execução, em vez de ser solicitada implicitamente e acessível implicitamente ou acidentalmente.

Criar bibliotecas compartilhadas com alinhamento ELF de 16 KB

Para criar bibliotecas compartilhadas que fazem parte do projeto Android, as configurações anteriores em Ativar tamanho de página de 16 KB são suficientes:

  • PRODUCT_NO_BIONIC_PAGE_SIZE_MACRO := true
  • PRODUCT_MAX_PAGE_SIZE_SUPPORTED := 16384

Para criar bibliotecas compartilhadas que não fazem parte do projeto do Android, é necessário transmitir esta flag do vinculador:

-Wl,-z,max-page-size=16384

Verificar binários e pré-criados para alinhamento ELF de 16 KB

A melhor maneira de verificar o alinhamento e o comportamento de execução é testar e executar em um kernel compilado de 16 KB. No entanto, para detectar alguns problemas mais cedo:

  • A partir do Android 16 (AOSP experimental), é possível definir PRODUCT_CHECK_PREBUILT_MAX_PAGE_SIZE := true no momento da criação. Use ignore_max_page_size: true em Android.bp e LOCAL_IGNORE_MAX_PAGE_SIZE := true em Android.mk para ignorá-los temporariamente. Essas configurações verificam todos os pré-criados e permitem detectar quando um é atualizado, mas não está alinhado a 16 KB.

  • É possível executar atest elf_alignment_test, que verifica o alinhamento de arquivos ELF no dispositivo em dispositivos lançados com o Android 15 e versões mais recentes.