| /* | 
 |  * Copyright (C) 2008 The Android Open Source Project | 
 |  * All rights reserved. | 
 |  * | 
 |  * Redistribution and use in source and binary forms, with or without | 
 |  * modification, are permitted provided that the following conditions | 
 |  * are met: | 
 |  *  * Redistributions of source code must retain the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer. | 
 |  *  * Redistributions in binary form must reproduce the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer in | 
 |  *    the documentation and/or other materials provided with the | 
 |  *    distribution. | 
 |  * | 
 |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
 |  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
 |  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
 |  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
 |  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
 |  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
 |  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS | 
 |  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | 
 |  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
 |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | 
 |  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | 
 |  * SUCH DAMAGE. | 
 |  */ | 
 |  | 
 | /* | 
 |  * This file is only used to provide backwards compatibility to property areas | 
 |  * created by old versions of init, which occurs when an ota runs.  The updater | 
 |  * binary is compiled statically against the newest bionic, but the recovery | 
 |  * ramdisk may be using an old version of init.  This can all be removed once | 
 |  * OTAs from pre-K versions are no longer supported. | 
 |  */ | 
 |  | 
 | #include <string.h> | 
 | #include <sys/atomics.h> | 
 |  | 
 | #define _REALLY_INCLUDE_SYS__SYSTEM_PROPERTIES_H_ | 
 | #include <sys/_system_properties.h> | 
 |  | 
 | #define TOC_NAME_LEN(toc)       ((toc) >> 24) | 
 | #define TOC_TO_INFO(area, toc)  ((prop_info_compat*) (((char*) area) + ((toc) & 0xFFFFFF))) | 
 |  | 
 | struct prop_area_compat { | 
 |     unsigned volatile count; | 
 |     unsigned volatile serial; | 
 |     unsigned magic; | 
 |     unsigned version; | 
 |     unsigned reserved[4]; | 
 |     unsigned toc[1]; | 
 | }; | 
 |  | 
 | typedef struct prop_area_compat prop_area_compat; | 
 |  | 
 | struct prop_area; | 
 | typedef struct prop_area prop_area; | 
 |  | 
 | struct prop_info_compat { | 
 |     char name[PROP_NAME_MAX]; | 
 |     unsigned volatile serial; | 
 |     char value[PROP_VALUE_MAX]; | 
 | }; | 
 |  | 
 | typedef struct prop_info_compat prop_info_compat; | 
 |  | 
 | extern prop_area *__system_property_area__; | 
 |  | 
 | const prop_info *__system_property_find_compat(const char *name) | 
 | { | 
 |     prop_area_compat *pa = (prop_area_compat *)__system_property_area__; | 
 |     unsigned count = pa->count; | 
 |     unsigned *toc = pa->toc; | 
 |     unsigned len = strlen(name); | 
 |     prop_info_compat *pi; | 
 |  | 
 |     if (len >= PROP_NAME_MAX) | 
 |         return 0; | 
 |     if (len < 1) | 
 |         return 0; | 
 |  | 
 |     while(count--) { | 
 |         unsigned entry = *toc++; | 
 |         if(TOC_NAME_LEN(entry) != len) continue; | 
 |  | 
 |         pi = TOC_TO_INFO(pa, entry); | 
 |         if(memcmp(name, pi->name, len)) continue; | 
 |  | 
 |         return (const prop_info *)pi; | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  | 
 | int __system_property_read_compat(const prop_info *_pi, char *name, char *value) | 
 | { | 
 |     unsigned serial, len; | 
 |     const prop_info_compat *pi = (const prop_info_compat *)_pi; | 
 |  | 
 |     for(;;) { | 
 |         serial = pi->serial; | 
 |         while(SERIAL_DIRTY(serial)) { | 
 |             __futex_wait((volatile void *)&pi->serial, serial, NULL); | 
 |             serial = pi->serial; | 
 |         } | 
 |         len = SERIAL_VALUE_LEN(serial); | 
 |         memcpy(value, pi->value, len + 1); | 
 |         if(serial == pi->serial) { | 
 |             if(name != 0) { | 
 |                 strcpy(name, pi->name); | 
 |             } | 
 |             return len; | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | int __system_property_foreach_compat( | 
 |         void (*propfn)(const prop_info *pi, void *cookie), | 
 |         void *cookie) | 
 | { | 
 |     prop_area_compat *pa = (prop_area_compat *)__system_property_area__; | 
 |     unsigned i; | 
 |  | 
 |     for (i = 0; i < pa->count; i++) { | 
 |         unsigned entry = pa->toc[i]; | 
 |         prop_info_compat *pi = TOC_TO_INFO(pa, entry); | 
 |         propfn((const prop_info *)pi, cookie); | 
 |     } | 
 |  | 
 |     return 0; | 
 | } |