1 /*
2  * Copyright (C) 2012 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "InputDevice"
18 
19 #include <stdlib.h>
20 #include <unistd.h>
21 #include <ctype.h>
22 
23 #include <input/InputDevice.h>
24 
25 namespace android {
26 
27 static const char* CONFIGURATION_FILE_DIR[] = {
28         "idc/",
29         "keylayout/",
30         "keychars/",
31 };
32 
33 static const char* CONFIGURATION_FILE_EXTENSION[] = {
34         ".idc",
35         ".kl",
36         ".kcm",
37 };
38 
isValidNameChar(char ch)39 static bool isValidNameChar(char ch) {
40     return isascii(ch) && (isdigit(ch) || isalpha(ch) || ch == '-' || ch == '_');
41 }
42 
appendInputDeviceConfigurationFileRelativePath(String8 & path,const String8 & name,InputDeviceConfigurationFileType type)43 static void appendInputDeviceConfigurationFileRelativePath(String8& path,
44         const String8& name, InputDeviceConfigurationFileType type) {
45     path.append(CONFIGURATION_FILE_DIR[type]);
46     for (size_t i = 0; i < name.length(); i++) {
47         char ch = name[i];
48         if (!isValidNameChar(ch)) {
49             ch = '_';
50         }
51         path.append(&ch, 1);
52     }
53     path.append(CONFIGURATION_FILE_EXTENSION[type]);
54 }
55 
getInputDeviceConfigurationFilePathByDeviceIdentifier(const InputDeviceIdentifier & deviceIdentifier,InputDeviceConfigurationFileType type)56 String8 getInputDeviceConfigurationFilePathByDeviceIdentifier(
57         const InputDeviceIdentifier& deviceIdentifier,
58         InputDeviceConfigurationFileType type) {
59     if (deviceIdentifier.vendor !=0 && deviceIdentifier.product != 0) {
60         if (deviceIdentifier.version != 0) {
61             // Try vendor product version.
62             String8 versionPath(getInputDeviceConfigurationFilePathByName(
63                     String8::format("Vendor_%04x_Product_%04x_Version_%04x",
64                             deviceIdentifier.vendor, deviceIdentifier.product,
65                             deviceIdentifier.version),
66                     type));
67             if (!versionPath.isEmpty()) {
68                 return versionPath;
69             }
70         }
71 
72         // Try vendor product.
73         String8 productPath(getInputDeviceConfigurationFilePathByName(
74                 String8::format("Vendor_%04x_Product_%04x",
75                         deviceIdentifier.vendor, deviceIdentifier.product),
76                 type));
77         if (!productPath.isEmpty()) {
78             return productPath;
79         }
80     }
81 
82     // Try device name.
83     return getInputDeviceConfigurationFilePathByName(deviceIdentifier.name, type);
84 }
85 
getInputDeviceConfigurationFilePathByName(const String8 & name,InputDeviceConfigurationFileType type)86 String8 getInputDeviceConfigurationFilePathByName(
87         const String8& name, InputDeviceConfigurationFileType type) {
88     // Search system repository.
89     String8 path;
90     path.setTo(getenv("ANDROID_ROOT"));
91     path.append("/usr/");
92     appendInputDeviceConfigurationFileRelativePath(path, name, type);
93 #if DEBUG_PROBE
94     ALOGD("Probing for system provided input device configuration file: path='%s'", path.string());
95 #endif
96     if (!access(path.string(), R_OK)) {
97 #if DEBUG_PROBE
98         ALOGD("Found");
99 #endif
100         return path;
101     }
102 
103     // Search user repository.
104     // TODO Should only look here if not in safe mode.
105     path.setTo(getenv("ANDROID_DATA"));
106     path.append("/system/devices/");
107     appendInputDeviceConfigurationFileRelativePath(path, name, type);
108 #if DEBUG_PROBE
109     ALOGD("Probing for system user input device configuration file: path='%s'", path.string());
110 #endif
111     if (!access(path.string(), R_OK)) {
112 #if DEBUG_PROBE
113         ALOGD("Found");
114 #endif
115         return path;
116     }
117 
118     // Not found.
119 #if DEBUG_PROBE
120     ALOGD("Probe failed to find input device configuration file: name='%s', type=%d",
121             name.string(), type);
122 #endif
123     return String8();
124 }
125 
126 
127 // --- InputDeviceInfo ---
128 
InputDeviceInfo()129 InputDeviceInfo::InputDeviceInfo() {
130     initialize(-1, 0, -1, InputDeviceIdentifier(), String8(), false, false);
131 }
132 
InputDeviceInfo(const InputDeviceInfo & other)133 InputDeviceInfo::InputDeviceInfo(const InputDeviceInfo& other) :
134         mId(other.mId), mGeneration(other.mGeneration), mControllerNumber(other.mControllerNumber),
135         mIdentifier(other.mIdentifier), mAlias(other.mAlias), mIsExternal(other.mIsExternal),
136         mHasMic(other.mHasMic), mSources(other.mSources),
137         mKeyboardType(other.mKeyboardType), mKeyCharacterMap(other.mKeyCharacterMap),
138         mHasVibrator(other.mHasVibrator), mHasButtonUnderPad(other.mHasButtonUnderPad),
139         mMotionRanges(other.mMotionRanges) {
140 }
141 
~InputDeviceInfo()142 InputDeviceInfo::~InputDeviceInfo() {
143 }
144 
initialize(int32_t id,int32_t generation,int32_t controllerNumber,const InputDeviceIdentifier & identifier,const String8 & alias,bool isExternal,bool hasMic)145 void InputDeviceInfo::initialize(int32_t id, int32_t generation, int32_t controllerNumber,
146         const InputDeviceIdentifier& identifier, const String8& alias, bool isExternal,
147         bool hasMic) {
148     mId = id;
149     mGeneration = generation;
150     mControllerNumber = controllerNumber;
151     mIdentifier = identifier;
152     mAlias = alias;
153     mIsExternal = isExternal;
154     mHasMic = hasMic;
155     mSources = 0;
156     mKeyboardType = AINPUT_KEYBOARD_TYPE_NONE;
157     mHasVibrator = false;
158     mHasButtonUnderPad = false;
159     mMotionRanges.clear();
160 }
161 
getMotionRange(int32_t axis,uint32_t source) const162 const InputDeviceInfo::MotionRange* InputDeviceInfo::getMotionRange(
163         int32_t axis, uint32_t source) const {
164     size_t numRanges = mMotionRanges.size();
165     for (size_t i = 0; i < numRanges; i++) {
166         const MotionRange& range = mMotionRanges.itemAt(i);
167         if (range.axis == axis && range.source == source) {
168             return &range;
169         }
170     }
171     return NULL;
172 }
173 
addSource(uint32_t source)174 void InputDeviceInfo::addSource(uint32_t source) {
175     mSources |= source;
176 }
177 
addMotionRange(int32_t axis,uint32_t source,float min,float max,float flat,float fuzz,float resolution)178 void InputDeviceInfo::addMotionRange(int32_t axis, uint32_t source, float min, float max,
179         float flat, float fuzz, float resolution) {
180     MotionRange range = { axis, source, min, max, flat, fuzz, resolution };
181     mMotionRanges.add(range);
182 }
183 
addMotionRange(const MotionRange & range)184 void InputDeviceInfo::addMotionRange(const MotionRange& range) {
185     mMotionRanges.add(range);
186 }
187 
188 } // namespace android
189