blob: 440b865a95e5e10c5fbec4f2e119852e89f5824a [file] [log] [blame]
Tom Cherryfd44b9f2017-11-08 14:01:00 -08001/*
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
29#include "context_node.h"
30
31#include <sys/mman.h>
32#include <unistd.h>
33
34#include <async_safe/log.h>
35
36#include "system_property_globals.h"
37
38// pthread_mutex_lock() calls into system_properties in the case of contention.
39// This creates a risk of dead lock if any system_properties functions
40// use pthread locks after system_property initialization.
41//
42// For this reason, the below three functions use a bionic Lock and static
43// allocation of memory for each filename.
44
45bool ContextNode::Open(bool access_rw, bool* fsetxattr_failed) {
46 lock_.lock();
47 if (pa_) {
48 lock_.unlock();
49 return true;
50 }
51
52 char filename[PROP_FILENAME_MAX];
53 int len =
54 async_safe_format_buffer(filename, sizeof(filename), "%s/%s", property_filename, context_);
55 if (len < 0 || len > PROP_FILENAME_MAX) {
56 lock_.unlock();
57 return false;
58 }
59
60 if (access_rw) {
61 pa_ = prop_area::map_prop_area_rw(filename, context_, fsetxattr_failed);
62 } else {
63 pa_ = prop_area::map_prop_area(filename);
64 }
65 lock_.unlock();
66 return pa_;
67}
68
69bool ContextNode::CheckAccessAndOpen() {
70 if (!pa_ && !no_access_) {
71 if (!CheckAccess() || !Open(false, nullptr)) {
72 no_access_ = true;
73 }
74 }
75 return pa_;
76}
77
78void ContextNode::ResetAccess() {
79 if (!CheckAccess()) {
80 Unmap();
81 no_access_ = true;
82 } else {
83 no_access_ = false;
84 }
85}
86
87bool ContextNode::CheckAccess() {
88 char filename[PROP_FILENAME_MAX];
89 int len =
90 async_safe_format_buffer(filename, sizeof(filename), "%s/%s", property_filename, context_);
91 if (len < 0 || len > PROP_FILENAME_MAX) {
92 return false;
93 }
94
95 return access(filename, R_OK) == 0;
96}
97
98void ContextNode::Unmap() {
99 if (!pa_) {
100 return;
101 }
102
103 munmap(pa_, pa_size);
104 pa_ = nullptr;
105}