• Home
  • History
  • Annotate
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1  /*
2   * Copyright (C) 2020 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  package com.android.commands.uinput;
18  
19  import android.annotation.Nullable;
20  import android.util.SparseArray;
21  
22  import java.util.Arrays;
23  import java.util.Objects;
24  
25  import src.com.android.commands.uinput.InputAbsInfo;
26  
27  /**
28   * An event is a JSON file defined action event to instruct uinput to perform a command like
29   * device registration or uinput events injection.
30   */
31  public class Event {
32      private static final String TAG = "UinputEvent";
33  
34      public enum Command {
35          REGISTER,
36          DELAY,
37          INJECT,
38          SYNC,
39          UPDATE_TIME_BASE,
40      }
41  
42      // Constants representing evdev event types, from include/uapi/linux/input-event-codes.h in the
43      // kernel.
44      public static final int EV_SYN = 0x00;
45      public static final int EV_KEY = 0x01;
46      public static final int EV_REL = 0x02;
47      public static final int EV_ABS = 0x03;
48      public static final int EV_MSC = 0x04;
49      public static final int EV_SW = 0x05;
50      public static final int EV_LED = 0x11;
51      public static final int EV_SND = 0x12;
52      public static final int EV_FF = 0x15;
53  
54      public enum UinputControlCode {
55          UI_SET_EVBIT(100),
56          UI_SET_KEYBIT(101),
57          UI_SET_RELBIT(102),
58          UI_SET_ABSBIT(103),
59          UI_SET_MSCBIT(104),
60          UI_SET_LEDBIT(105),
61          UI_SET_SNDBIT(106),
62          UI_SET_FFBIT(107),
63          UI_SET_SWBIT(109),
64          UI_SET_PROPBIT(110);
65  
66          private final int mValue;
67  
UinputControlCode(int value)68          UinputControlCode(int value) {
69              this.mValue = value;
70          }
71  
getValue()72          public int getValue() {
73              return mValue;
74          }
75  
76          /**
77           * Returns the control code for the given evdev event type, or {@code null} if there is no
78           * control code for that type.
79           */
forEventType(int eventType)80          public static @Nullable UinputControlCode forEventType(int eventType) {
81              return switch (eventType) {
82                  case EV_KEY -> UI_SET_KEYBIT;
83                  case EV_REL -> UI_SET_RELBIT;
84                  case EV_ABS -> UI_SET_ABSBIT;
85                  case EV_MSC -> UI_SET_MSCBIT;
86                  case EV_SW -> UI_SET_SWBIT;
87                  case EV_LED -> UI_SET_LEDBIT;
88                  case EV_SND -> UI_SET_SNDBIT;
89                  case EV_FF -> UI_SET_FFBIT;
90                  default -> null;
91              };
92          }
93      }
94  
95      private int mId;
96      private Command mCommand;
97      private String mName;
98      private int mVendorId;
99      private int mProductId;
100      private int mVersionId;
101      private int mBusId;
102      private int[] mInjections;
103      private long mTimestampOffsetMicros = -1;
104      private SparseArray<int[]> mConfiguration;
105      private long mDurationNanos;
106      private int mFfEffectsMax = 0;
107      private String mInputPort;
108      private SparseArray<InputAbsInfo> mAbsInfo;
109      private String mSyncToken;
110  
getId()111      public int getId() {
112          return mId;
113      }
114  
getCommand()115      public Command getCommand() {
116          return mCommand;
117      }
118  
getName()119      public String getName() {
120          return mName;
121      }
122  
getVendorId()123      public int getVendorId() {
124          return mVendorId;
125      }
126  
getProductId()127      public int getProductId() {
128          return mProductId;
129      }
130  
getVersionId()131      public int getVersionId() {
132          return mVersionId;
133      }
134  
getBus()135      public int getBus() {
136          return mBusId;
137      }
138  
getInjections()139      public int[] getInjections() {
140          return mInjections;
141      }
142  
143      /**
144       * Returns the number of microseconds that should be added to the previous {@code INJECT}
145       * event's timestamp to produce the timestamp for this {@code INJECT} event. A value of -1
146       * indicates that the current timestamp should be used instead.
147       */
getTimestampOffsetMicros()148      public long getTimestampOffsetMicros() {
149          return mTimestampOffsetMicros;
150      }
151  
152      /**
153       * Returns a {@link SparseArray} describing the event codes that should be registered for the
154       * device. The keys are uinput ioctl codes (such as those returned from {@link
155       * UinputControlCode#getValue()}, while the values are arrays of event codes to be enabled with
156       * those ioctls. For example, key 101 (corresponding to {@link UinputControlCode#UI_SET_KEYBIT})
157       * could have values 0x110 ({@code BTN_LEFT}), 0x111 ({@code BTN_RIGHT}), and 0x112
158       * ({@code BTN_MIDDLE}).
159       */
getConfiguration()160      public SparseArray<int[]> getConfiguration() {
161          return mConfiguration;
162      }
163  
getDurationNanos()164      public long getDurationNanos() {
165          return mDurationNanos;
166      }
167  
getFfEffectsMax()168      public int getFfEffectsMax() {
169          return mFfEffectsMax;
170      }
171  
getAbsInfo()172      public SparseArray<InputAbsInfo>  getAbsInfo() {
173          return mAbsInfo;
174      }
175  
getPort()176      public String getPort() {
177          return mInputPort;
178      }
179  
getSyncToken()180      public String getSyncToken() {
181          return mSyncToken;
182      }
183  
184      /**
185       * Convert an event to String.
186       */
toString()187      public String toString() {
188          return "Event{id=" + mId
189              + ", command=" + mCommand
190              + ", name=" + mName
191              + ", vid=" + mVendorId
192              + ", pid=" + mProductId
193              + ", busId=" + mBusId
194              + ", events=" + Arrays.toString(mInjections)
195              + ", configuration=" + mConfiguration
196              + ", duration=" + mDurationNanos + "ns"
197              + ", ff_effects_max=" + mFfEffectsMax
198              + ", port=" + mInputPort
199              + "}";
200      }
201  
202      public static class Builder {
203          private Event mEvent;
204  
Builder()205          Builder() {
206              mEvent = new Event();
207          }
208  
setId(int id)209          public void setId(int id) {
210              mEvent.mId = id;
211          }
212  
setCommand(Command command)213          public void setCommand(Command command) {
214              mEvent.mCommand = command;
215          }
216  
setName(String name)217          public void setName(String name) {
218              mEvent.mName = name;
219          }
220  
setInjections(int[] events)221          public void setInjections(int[] events) {
222              mEvent.mInjections = events;
223          }
224  
setTimestampOffsetMicros(long offsetMicros)225          public void setTimestampOffsetMicros(long offsetMicros) {
226              mEvent.mTimestampOffsetMicros = offsetMicros;
227          }
228  
229          /**
230           * Sets the event codes that should be registered with a {@code register} command.
231           *
232           * @param configuration An array of ioctls and event codes, as described at
233           *                      {@link Event#getConfiguration()}.
234           */
setConfiguration(SparseArray<int[]> configuration)235          public void setConfiguration(SparseArray<int[]> configuration) {
236              mEvent.mConfiguration = configuration;
237          }
238  
setVendorId(int vendorId)239          public void setVendorId(int vendorId) {
240              mEvent.mVendorId = vendorId;
241          }
242  
setProductId(int productId)243          public void setProductId(int productId) {
244              mEvent.mProductId = productId;
245          }
246  
setVersionId(int versionId)247          public void setVersionId(int versionId) {
248              mEvent.mVersionId = versionId;
249          }
250  
setBusId(int busId)251          public void setBusId(int busId) {
252              mEvent.mBusId = busId;
253          }
254  
setDurationNanos(long durationNanos)255          public void setDurationNanos(long durationNanos) {
256              mEvent.mDurationNanos = durationNanos;
257          }
258  
setFfEffectsMax(int ffEffectsMax)259          public void setFfEffectsMax(int ffEffectsMax) {
260              mEvent.mFfEffectsMax = ffEffectsMax;
261          }
262  
setAbsInfo(SparseArray<InputAbsInfo> absInfo)263          public void setAbsInfo(SparseArray<InputAbsInfo> absInfo) {
264              mEvent.mAbsInfo = absInfo;
265          }
266  
setInputPort(String port)267          public void setInputPort(String port) {
268              mEvent.mInputPort = port;
269          }
270  
setSyncToken(String syncToken)271          public void setSyncToken(String syncToken) {
272              mEvent.mSyncToken = Objects.requireNonNull(syncToken, "Sync token must not be null");
273          }
274  
build()275          public Event build() {
276              if (mEvent.mId == -1) {
277                  throw new IllegalStateException("No event id");
278              } else if (mEvent.mCommand == null) {
279                  throw new IllegalStateException("Event does not contain a command");
280              }
281              switch (mEvent.mCommand) {
282                  case REGISTER -> {
283                      if (mEvent.mConfiguration == null) {
284                          throw new IllegalStateException(
285                                  "Device registration is missing configuration");
286                      }
287                  }
288                  case DELAY -> {
289                      if (mEvent.mDurationNanos <= 0) {
290                          throw new IllegalStateException("Delay has missing or invalid duration");
291                      }
292                  }
293                  case INJECT -> {
294                      if (mEvent.mInjections == null) {
295                          throw new IllegalStateException("Inject command is missing injection data");
296                      }
297                  }
298                  case SYNC -> {
299                      if (mEvent.mSyncToken == null) {
300                          throw new IllegalStateException("Sync command is missing sync token");
301                      }
302                  }
303              }
304              return mEvent;
305          }
306      }
307  }
308