xref: /aosp_15_r20/bionic/libc/private/KernelArgumentBlock.h (revision 8d67ca893c1523eb926b9080dbe4e2ffd2a27ba1)
1*8d67ca89SAndroid Build Coastguard Worker /*
2*8d67ca89SAndroid Build Coastguard Worker  * Copyright (C) 2013 The Android Open Source Project
3*8d67ca89SAndroid Build Coastguard Worker  *
4*8d67ca89SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*8d67ca89SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*8d67ca89SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*8d67ca89SAndroid Build Coastguard Worker  *
8*8d67ca89SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*8d67ca89SAndroid Build Coastguard Worker  *
10*8d67ca89SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*8d67ca89SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*8d67ca89SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*8d67ca89SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*8d67ca89SAndroid Build Coastguard Worker  * limitations under the License.
15*8d67ca89SAndroid Build Coastguard Worker  */
16*8d67ca89SAndroid Build Coastguard Worker 
17*8d67ca89SAndroid Build Coastguard Worker #pragma once
18*8d67ca89SAndroid Build Coastguard Worker 
19*8d67ca89SAndroid Build Coastguard Worker #include <elf.h>
20*8d67ca89SAndroid Build Coastguard Worker #include <link.h>
21*8d67ca89SAndroid Build Coastguard Worker #include <stdint.h>
22*8d67ca89SAndroid Build Coastguard Worker #include <sys/auxv.h>
23*8d67ca89SAndroid Build Coastguard Worker 
24*8d67ca89SAndroid Build Coastguard Worker #include "platform/bionic/macros.h"
25*8d67ca89SAndroid Build Coastguard Worker 
26*8d67ca89SAndroid Build Coastguard Worker // When the kernel starts the dynamic linker, it passes a pointer to a block
27*8d67ca89SAndroid Build Coastguard Worker // of memory containing argc, the argv array, the environment variable array,
28*8d67ca89SAndroid Build Coastguard Worker // and the array of ELF aux vectors. This class breaks that block up into its
29*8d67ca89SAndroid Build Coastguard Worker // constituents for easy access.
30*8d67ca89SAndroid Build Coastguard Worker class KernelArgumentBlock {
31*8d67ca89SAndroid Build Coastguard Worker  public:
KernelArgumentBlock(void * raw_args)32*8d67ca89SAndroid Build Coastguard Worker   __attribute__((no_sanitize("hwaddress"))) explicit KernelArgumentBlock(void* raw_args) {
33*8d67ca89SAndroid Build Coastguard Worker     uintptr_t* args = reinterpret_cast<uintptr_t*>(raw_args);
34*8d67ca89SAndroid Build Coastguard Worker     argc = static_cast<int>(*args);
35*8d67ca89SAndroid Build Coastguard Worker     argv = reinterpret_cast<char**>(args + 1);
36*8d67ca89SAndroid Build Coastguard Worker     envp = argv + argc + 1;
37*8d67ca89SAndroid Build Coastguard Worker 
38*8d67ca89SAndroid Build Coastguard Worker     // Skip over all environment variable definitions to find the aux vector.
39*8d67ca89SAndroid Build Coastguard Worker     // The end of the environment block is marked by a NULL pointer.
40*8d67ca89SAndroid Build Coastguard Worker     char** p = envp;
41*8d67ca89SAndroid Build Coastguard Worker     while (*p != nullptr) {
42*8d67ca89SAndroid Build Coastguard Worker       ++p;
43*8d67ca89SAndroid Build Coastguard Worker     }
44*8d67ca89SAndroid Build Coastguard Worker     ++p; // Skip the NULL itself.
45*8d67ca89SAndroid Build Coastguard Worker 
46*8d67ca89SAndroid Build Coastguard Worker     auxv = reinterpret_cast<ElfW(auxv_t)*>(p);
47*8d67ca89SAndroid Build Coastguard Worker   }
48*8d67ca89SAndroid Build Coastguard Worker 
49*8d67ca89SAndroid Build Coastguard Worker   // Similar to ::getauxval but doesn't require the libc global variables to be set up,
50*8d67ca89SAndroid Build Coastguard Worker   // so it's safe to call this really early on.
getauxval(unsigned long type)51*8d67ca89SAndroid Build Coastguard Worker   __attribute__((no_sanitize("hwaddress"))) unsigned long getauxval(unsigned long type) {
52*8d67ca89SAndroid Build Coastguard Worker     for (ElfW(auxv_t)* v = auxv; v->a_type != AT_NULL; ++v) {
53*8d67ca89SAndroid Build Coastguard Worker       if (v->a_type == type) {
54*8d67ca89SAndroid Build Coastguard Worker         return v->a_un.a_val;
55*8d67ca89SAndroid Build Coastguard Worker       }
56*8d67ca89SAndroid Build Coastguard Worker     }
57*8d67ca89SAndroid Build Coastguard Worker     return 0;
58*8d67ca89SAndroid Build Coastguard Worker   }
59*8d67ca89SAndroid Build Coastguard Worker 
60*8d67ca89SAndroid Build Coastguard Worker   int argc;
61*8d67ca89SAndroid Build Coastguard Worker   char** argv;
62*8d67ca89SAndroid Build Coastguard Worker   char** envp;
63*8d67ca89SAndroid Build Coastguard Worker   ElfW(auxv_t)* auxv;
64*8d67ca89SAndroid Build Coastguard Worker 
65*8d67ca89SAndroid Build Coastguard Worker  private:
66*8d67ca89SAndroid Build Coastguard Worker   BIONIC_DISALLOW_COPY_AND_ASSIGN(KernelArgumentBlock);
67*8d67ca89SAndroid Build Coastguard Worker };
68