1 /*
2 * Copyright (C) 2008 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28 #include <grp.h>
29 #include <stdio.h>
30 #include <unistd.h>
31 #include <pwd.h>
32 #include <netdb.h>
33 #include <mntent.h>
34 #include <private/android_filesystem_config.h>
35 #include <pthread.h>
36 #include <stdlib.h>
37 #include <errno.h>
38 #include <ctype.h>
39
40 /** Thread-specific state for the stubs functions
41 **/
42
43 pthread_once_t the_once = PTHREAD_ONCE_INIT;
44 pthread_key_t the_key;
45
46 typedef struct {
47 struct passwd passwd;
48 struct group group;
49 char* group_members[2];
50 char app_name_buffer[32];
51 char group_name_buffer[32];
52 } stubs_state_t;
53
54 static void
stubs_state_free(void * _s)55 stubs_state_free( void* _s )
56 {
57 stubs_state_t* s = _s;
58 free(s);
59 }
60
61 static stubs_state_t*
stubs_state_alloc(void)62 stubs_state_alloc( void )
63 {
64 stubs_state_t* s = calloc(1, sizeof *s);
65
66 if (s != NULL) {
67 s->group.gr_mem = s->group_members;
68 }
69 return s;
70 }
71
__stubs_key_init(void)72 static void __stubs_key_init(void)
73 {
74 pthread_key_create( &the_key, stubs_state_free );
75 }
76
77 static stubs_state_t*
__stubs_state(void)78 __stubs_state(void)
79 {
80 stubs_state_t* s;
81
82 pthread_once(&the_once, __stubs_key_init);
83 s = pthread_getspecific(the_key);
84 if (s == NULL) {
85 s = stubs_state_alloc();
86 if (s == NULL) {
87 errno = ENOMEM; /* just in case */
88 } else {
89 if ( pthread_setspecific(the_key, s) != 0 ) {
90 stubs_state_free(s);
91 errno = ENOMEM;
92 s = NULL;
93 }
94 }
95 }
96 return s;
97 }
98
99 static struct passwd*
android_iinfo_to_passwd(struct passwd * pw,struct android_id_info * iinfo)100 android_iinfo_to_passwd( struct passwd *pw,
101 struct android_id_info *iinfo )
102 {
103 pw->pw_name = (char*)iinfo->name;
104 pw->pw_uid = iinfo->aid;
105 pw->pw_gid = iinfo->aid;
106 pw->pw_dir = "/";
107 pw->pw_shell = "/system/bin/sh";
108 return pw;
109 }
110
111 static struct group*
android_iinfo_to_group(struct group * gr,struct android_id_info * iinfo)112 android_iinfo_to_group( struct group *gr,
113 struct android_id_info *iinfo )
114 {
115 gr->gr_name = (char*) iinfo->name;
116 gr->gr_gid = iinfo->aid;
117 gr->gr_mem[0] = gr->gr_name;
118 gr->gr_mem[1] = NULL;
119 return gr;
120 }
121
122 static struct passwd *
android_id_to_passwd(struct passwd * pw,unsigned id)123 android_id_to_passwd( struct passwd *pw, unsigned id)
124 {
125 struct android_id_info *iinfo = android_ids;
126 unsigned n;
127 for (n = 0; n < android_id_count; n++) {
128 if (iinfo[n].aid == id) {
129 return android_iinfo_to_passwd(pw, iinfo + n);
130 }
131 }
132 return NULL;
133 }
134
135 static struct passwd*
android_name_to_passwd(struct passwd * pw,const char * name)136 android_name_to_passwd(struct passwd *pw, const char *name)
137 {
138 struct android_id_info *iinfo = android_ids;
139 unsigned n;
140 for (n = 0; n < android_id_count; n++) {
141 if (!strcmp(iinfo[n].name, name)) {
142 return android_iinfo_to_passwd(pw, iinfo + n);
143 }
144 }
145 return NULL;
146 }
147
148 static struct group*
android_id_to_group(struct group * gr,unsigned id)149 android_id_to_group( struct group *gr, unsigned id )
150 {
151 struct android_id_info *iinfo = android_ids;
152 unsigned n;
153 for (n = 0; n < android_id_count; n++) {
154 if (iinfo[n].aid == id) {
155 return android_iinfo_to_group(gr, iinfo + n);
156 }
157 }
158 return NULL;
159 }
160
161 static struct group*
android_name_to_group(struct group * gr,const char * name)162 android_name_to_group( struct group *gr, const char *name )
163 {
164 struct android_id_info *iinfo = android_ids;
165 unsigned n;
166 for (n = 0; n < android_id_count; n++) {
167 if (!strcmp(iinfo[n].name, name)) {
168 return android_iinfo_to_group(gr, iinfo + n);
169 }
170 }
171 return NULL;
172 }
173
174 /* translate a user/group name like app_1234 into the
175 * corresponding user/group id (AID_APP + 1234)
176 * returns 0 and sets errno to ENOENT in case of error
177 */
178 static unsigned
app_id_from_name(const char * name)179 app_id_from_name( const char* name )
180 {
181 unsigned long id;
182 char* end;
183
184 if (memcmp(name, "app_", 4) != 0 || !isdigit(name[4]))
185 goto FAIL;
186
187 id = strtoul(name+4, &end, 10);
188 if (*end != '\0')
189 goto FAIL;
190
191 id += AID_APP;
192
193 /* check for overflow and that the value can be
194 * stored in our 32-bit uid_t/gid_t */
195 if (id < AID_APP || (unsigned)id != id)
196 goto FAIL;
197
198 return (unsigned)id;
199
200 FAIL:
201 errno = ENOENT;
202 return 0;
203 }
204
205 /* translate a uid into the corresponding app_<uid>
206 * passwd structure (sets errno to ENOENT on failure)
207 */
208 static struct passwd*
app_id_to_passwd(uid_t uid,stubs_state_t * state)209 app_id_to_passwd(uid_t uid, stubs_state_t* state)
210 {
211 struct passwd* pw = &state->passwd;
212
213 if (uid < AID_APP) {
214 errno = ENOENT;
215 return NULL;
216 }
217
218 snprintf( state->app_name_buffer, sizeof state->app_name_buffer,
219 "app_%u", uid - AID_APP );
220
221 pw->pw_name = state->app_name_buffer;
222 pw->pw_dir = "/data";
223 pw->pw_shell = "/system/bin/sh";
224 pw->pw_uid = uid;
225 pw->pw_gid = uid;
226
227 return pw;
228 }
229
230 /* translate a gid into the corresponding app_<gid>
231 * group structure (sets errno to ENOENT on failure)
232 */
233 static struct group*
app_id_to_group(gid_t gid,stubs_state_t * state)234 app_id_to_group(gid_t gid, stubs_state_t* state)
235 {
236 struct group* gr = &state->group;
237
238 if (gid < AID_APP) {
239 errno = ENOENT;
240 return NULL;
241 }
242
243 snprintf(state->group_name_buffer, sizeof state->group_name_buffer,
244 "app_%u", gid - AID_APP);
245
246 gr->gr_name = state->group_name_buffer;
247 gr->gr_gid = gid;
248 gr->gr_mem[0] = gr->gr_name;
249 gr->gr_mem[1] = NULL;
250
251 return gr;
252 }
253
254
255 struct passwd*
getpwuid(uid_t uid)256 getpwuid(uid_t uid)
257 {
258 stubs_state_t* state = __stubs_state();
259 struct passwd* pw;
260
261 if (state == NULL)
262 return NULL;
263
264 pw = &state->passwd;
265
266 if ( android_id_to_passwd(pw, uid) != NULL )
267 return pw;
268
269 return app_id_to_passwd(uid, state);
270 }
271
272 struct passwd*
getpwnam(const char * login)273 getpwnam(const char *login)
274 {
275 stubs_state_t* state = __stubs_state();
276
277 if (state == NULL)
278 return NULL;
279
280 if (android_name_to_passwd(&state->passwd, login) != NULL)
281 return &state->passwd;
282
283 return app_id_to_passwd( app_id_from_name(login), state );
284 }
285
286 int
getgrouplist(const char * user,gid_t group,gid_t * groups,int * ngroups)287 getgrouplist (const char *user, gid_t group,
288 gid_t *groups, int *ngroups)
289 {
290 if (*ngroups < 1) {
291 *ngroups = 1;
292 return -1;
293 }
294 groups[0] = group;
295 return (*ngroups = 1);
296 }
297
298 char*
getlogin(void)299 getlogin(void)
300 {
301 struct passwd *pw = getpwuid(getuid());
302
303 if(pw) {
304 return pw->pw_name;
305 } else {
306 return NULL;
307 }
308 }
309
310 struct group*
getgrgid(gid_t gid)311 getgrgid(gid_t gid)
312 {
313 stubs_state_t* state = __stubs_state();
314 struct group* gr;
315
316 if (state == NULL)
317 return NULL;
318
319 gr = android_id_to_group(&state->group, gid);
320 if (gr != NULL)
321 return gr;
322
323 return app_id_to_group(gid, state);
324 }
325
326 struct group*
getgrnam(const char * name)327 getgrnam(const char *name)
328 {
329 stubs_state_t* state = __stubs_state();
330 unsigned id;
331
332 if (state == NULL)
333 return NULL;
334
335 if (android_name_to_group(&state->group, name) != 0)
336 return &state->group;
337
338 return app_id_to_group( app_id_from_name(name), state );
339 }
340
341
getnetbyname(const char * name)342 struct netent* getnetbyname(const char *name)
343 {
344 fprintf(stderr, "FIX ME! implement getgrnam() %s:%d\n", __FILE__, __LINE__);
345 return NULL;
346 }
347
endpwent(void)348 void endpwent(void)
349 {
350 }
351
getmntent(FILE * f)352 struct mntent* getmntent(FILE* f)
353 {
354 fprintf(stderr, "FIX ME! implement getmntent() %s:%d\n", __FILE__, __LINE__);
355 return NULL;
356 }
357
ttyname(int fd)358 char* ttyname(int fd)
359 {
360 fprintf(stderr, "FIX ME! implement ttyname() %s:%d\n", __FILE__, __LINE__);
361 return NULL;
362 }
363
ttyname_r(int fd,char * buf,size_t buflen)364 int ttyname_r(int fd, char *buf, size_t buflen)
365 {
366 fprintf(stderr, "FIX ME! implement ttyname_r() %s:%d\n", __FILE__, __LINE__);
367 return -ERANGE;
368 }
369
getnetbyaddr(uint32_t net,int type)370 struct netent *getnetbyaddr(uint32_t net, int type)
371 {
372 fprintf(stderr, "FIX ME! implement %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
373 return NULL;
374 }
375
getprotobyname(const char * name)376 struct protoent *getprotobyname(const char *name)
377 {
378 fprintf(stderr, "FIX ME! implement %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
379 return NULL;
380 }
381
getprotobynumber(int proto)382 struct protoent *getprotobynumber(int proto)
383 {
384 fprintf(stderr, "FIX ME! implement %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
385 return NULL;
386 }
387
getusershell(void)388 char* getusershell(void)
389 {
390 fprintf(stderr, "FIX ME! implement %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
391 return NULL;
392 }
393
setusershell(void)394 void setusershell(void)
395 {
396 fprintf(stderr, "FIX ME! implement %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
397 }
398
endusershell(void)399 void endusershell(void)
400 {
401 fprintf(stderr, "FIX ME! implement %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
402 }
403
404