blob: f0c81c9d8e23a49b404c16ea3e2b95a0912afd6e [file] [log] [blame]
Elliott Hughes9e27e582017-03-23 17:42:49 -07001/*
2 * Copyright (C) 2017 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
29#pragma once
30
31#include <string.h>
32
33#define _REALLY_INCLUDE_SYS__SYSTEM_PROPERTIES_H_
34#include <sys/_system_properties.h>
35
Elliott Hughesa98aa562017-04-11 17:23:37 -070036// Cached system property lookup. For code that needs to read the same property multiple times,
37// this class helps optimize those lookups.
Elliott Hughes9e27e582017-03-23 17:42:49 -070038class CachedProperty {
39 public:
Elliott Hughesa98aa562017-04-11 17:23:37 -070040 // The lifetime of `property_name` must be greater than that of this CachedProperty.
Elliott Hughes9e27e582017-03-23 17:42:49 -070041 CachedProperty(const char* property_name)
42 : property_name_(property_name),
43 prop_info_(nullptr),
44 cached_area_serial_(0),
45 cached_property_serial_(0) {
46 cached_value_[0] = '\0';
47 }
48
Elliott Hughesa98aa562017-04-11 17:23:37 -070049 // Returns the current value of the underlying system property as cheaply as possible.
50 // The returned pointer is valid until the next call to Get. It is the caller's responsibility
51 // to provide a lock for thread-safety.
Elliott Hughes9e27e582017-03-23 17:42:49 -070052 const char* Get() {
Elliott Hughes9e27e582017-03-23 17:42:49 -070053 // Do we have a `struct prop_info` yet?
54 if (prop_info_ == nullptr) {
55 // `__system_property_find` is expensive, so only retry if a property
56 // has been created since last time we checked.
57 uint32_t property_area_serial = __system_property_area_serial();
58 if (property_area_serial != cached_area_serial_) {
59 prop_info_ = __system_property_find(property_name_);
60 cached_area_serial_ = property_area_serial;
61 }
62 }
63
64 if (prop_info_ != nullptr) {
65 // Only bother re-reading the property if it's actually changed since last time.
66 uint32_t property_serial = __system_property_serial(prop_info_);
67 if (property_serial != cached_property_serial_) {
68 __system_property_read_callback(prop_info_, &CachedProperty::Callback, this);
69 }
70 }
71
Elliott Hughes9e27e582017-03-23 17:42:49 -070072 return cached_value_;
73 }
74
75 private:
Elliott Hughes9e27e582017-03-23 17:42:49 -070076 const char* property_name_;
77 const prop_info* prop_info_;
78 uint32_t cached_area_serial_;
79 uint32_t cached_property_serial_;
80 char cached_value_[PROP_VALUE_MAX];
81
82 static void Callback(void* data, const char*, const char* value, uint32_t serial) {
83 CachedProperty* instance = reinterpret_cast<CachedProperty*>(data);
84 instance->cached_property_serial_ = serial;
85 strcpy(instance->cached_value_, value);
86 }
87};