1 /* 2 * Copyright (C) 2014 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 package com.android.internal.alsa; 17 18 import android.util.Slog; 19 import java.io.BufferedReader; 20 import java.io.File; 21 import java.io.FileNotFoundException; 22 import java.io.FileReader; 23 import java.io.IOException; 24 import java.util.ArrayList; 25 26 /** 27 * @hide 28 * Retrieves information from an ALSA "devices" file. 29 */ 30 public class AlsaDevicesParser { 31 private static final String TAG = "AlsaDevicesParser"; 32 protected static final boolean DEBUG = false; 33 34 private static final String kDevicesFilePath = "/proc/asound/devices"; 35 36 private static final int kIndex_CardDeviceField = 5; 37 private static final int kStartIndex_CardNum = 6; 38 private static final int kEndIndex_CardNum = 8; // one past 39 private static final int kStartIndex_DeviceNum = 9; 40 private static final int kEndIndex_DeviceNum = 11; // one past 41 private static final int kStartIndex_Type = 14; 42 43 private static LineTokenizer mTokenizer = new LineTokenizer(" :[]-"); 44 45 private boolean mHasCaptureDevices = false; 46 private boolean mHasPlaybackDevices = false; 47 private boolean mHasMIDIDevices = false; 48 49 public class AlsaDeviceRecord { 50 public static final int kDeviceType_Unknown = -1; 51 public static final int kDeviceType_Audio = 0; 52 public static final int kDeviceType_Control = 1; 53 public static final int kDeviceType_MIDI = 2; 54 55 public static final int kDeviceDir_Unknown = -1; 56 public static final int kDeviceDir_Capture = 0; 57 public static final int kDeviceDir_Playback = 1; 58 59 int mCardNum = -1; 60 int mDeviceNum = -1; 61 int mDeviceType = kDeviceType_Unknown; 62 int mDeviceDir = kDeviceDir_Unknown; 63 AlsaDeviceRecord()64 public AlsaDeviceRecord() {} 65 parse(String line)66 public boolean parse(String line) { 67 // "0123456789012345678901234567890" 68 // " 2: [ 0-31]: digital audio playback" 69 // " 3: [ 0-30]: digital audio capture" 70 // " 35: [ 1] : control" 71 // " 36: [ 2- 0]: raw midi" 72 73 final int kToken_LineNum = 0; 74 final int kToken_CardNum = 1; 75 final int kToken_DeviceNum = 2; 76 final int kToken_Type0 = 3; // "digital", "control", "raw" 77 final int kToken_Type1 = 4; // "audio", "midi" 78 final int kToken_Type2 = 5; // "capture", "playback" 79 80 int tokenOffset = 0; 81 int delimOffset = 0; 82 int tokenIndex = kToken_LineNum; 83 while (true) { 84 tokenOffset = mTokenizer.nextToken(line, delimOffset); 85 if (tokenOffset == LineTokenizer.kTokenNotFound) { 86 break; // bail 87 } 88 delimOffset = mTokenizer.nextDelimiter(line, tokenOffset); 89 if (delimOffset == LineTokenizer.kTokenNotFound) { 90 delimOffset = line.length(); 91 } 92 String token = line.substring(tokenOffset, delimOffset); 93 94 try { 95 switch (tokenIndex) { 96 case kToken_LineNum: 97 // ignore 98 break; 99 100 case kToken_CardNum: 101 mCardNum = Integer.parseInt(token); 102 if (line.charAt(delimOffset) != '-') { 103 tokenIndex++; // no device # in the token stream 104 } 105 break; 106 107 case kToken_DeviceNum: 108 mDeviceNum = Integer.parseInt(token); 109 break; 110 111 case kToken_Type0: 112 if (token.equals("digital")) { 113 // NOP 114 } else if (token.equals("control")) { 115 mDeviceType = kDeviceType_Control; 116 } else if (token.equals("raw")) { 117 // NOP 118 } 119 break; 120 121 case kToken_Type1: 122 if (token.equals("audio")) { 123 mDeviceType = kDeviceType_Audio; 124 } else if (token.equals("midi")) { 125 mDeviceType = kDeviceType_MIDI; 126 mHasMIDIDevices = true; 127 } 128 break; 129 130 case kToken_Type2: 131 if (token.equals("capture")) { 132 mDeviceDir = kDeviceDir_Capture; 133 mHasCaptureDevices = true; 134 } else if (token.equals("playback")) { 135 mDeviceDir = kDeviceDir_Playback; 136 mHasPlaybackDevices = true; 137 } 138 break; 139 } // switch (tokenIndex) 140 } catch (NumberFormatException e) { 141 Slog.e(TAG, "Failed to parse token " + tokenIndex + " of " + kDevicesFilePath 142 + " token: " + token); 143 return false; 144 } 145 146 tokenIndex++; 147 } // while (true) 148 149 return true; 150 } // parse() 151 textFormat()152 public String textFormat() { 153 StringBuilder sb = new StringBuilder(); 154 sb.append("[" + mCardNum + ":" + mDeviceNum + "]"); 155 156 switch (mDeviceType) { 157 case kDeviceType_Unknown: 158 sb.append(" N/A"); 159 break; 160 case kDeviceType_Audio: 161 sb.append(" Audio"); 162 break; 163 case kDeviceType_Control: 164 sb.append(" Control"); 165 break; 166 case kDeviceType_MIDI: 167 sb.append(" MIDI"); 168 break; 169 } 170 171 switch (mDeviceDir) { 172 case kDeviceDir_Unknown: 173 sb.append(" N/A"); 174 break; 175 case kDeviceDir_Capture: 176 sb.append(" Capture"); 177 break; 178 case kDeviceDir_Playback: 179 sb.append(" Playback"); 180 break; 181 } 182 183 return sb.toString(); 184 } 185 } 186 187 private final ArrayList<AlsaDeviceRecord> mDeviceRecords = new ArrayList<AlsaDeviceRecord>(); 188 AlsaDevicesParser()189 public AlsaDevicesParser() {} 190 191 // 192 // Access 193 // getDefaultDeviceNum(int card)194 public int getDefaultDeviceNum(int card) { 195 // TODO - This (obviously) isn't sufficient. Revisit. 196 return 0; 197 } 198 199 // 200 // Predicates 201 // 202 /* 203 public boolean hasPlaybackDevices() { 204 return mHasPlaybackDevices; 205 } 206 */ 207 hasPlaybackDevices(int card)208 public boolean hasPlaybackDevices(int card) { 209 for (AlsaDeviceRecord deviceRecord : mDeviceRecords) { 210 if (deviceRecord.mCardNum == card && 211 deviceRecord.mDeviceType == AlsaDeviceRecord.kDeviceType_Audio && 212 deviceRecord.mDeviceDir == AlsaDeviceRecord.kDeviceDir_Playback) { 213 return true; 214 } 215 } 216 return false; 217 } 218 219 /* 220 public boolean hasCaptureDevices() { 221 return mHasCaptureDevices; 222 } 223 */ 224 hasCaptureDevices(int card)225 public boolean hasCaptureDevices(int card) { 226 for (AlsaDeviceRecord deviceRecord : mDeviceRecords) { 227 if (deviceRecord.mCardNum == card && 228 deviceRecord.mDeviceType == AlsaDeviceRecord.kDeviceType_Audio && 229 deviceRecord.mDeviceDir == AlsaDeviceRecord.kDeviceDir_Capture) { 230 return true; 231 } 232 } 233 return false; 234 } 235 236 /* 237 public boolean hasMIDIDevices() { 238 return mHasMIDIDevices; 239 } 240 */ 241 hasMIDIDevices(int card)242 public boolean hasMIDIDevices(int card) { 243 for (AlsaDeviceRecord deviceRecord : mDeviceRecords) { 244 if (deviceRecord.mCardNum == card && 245 deviceRecord.mDeviceType == AlsaDeviceRecord.kDeviceType_MIDI) { 246 return true; 247 } 248 } 249 return false; 250 } 251 252 // 253 // Process 254 // isLineDeviceRecord(String line)255 private boolean isLineDeviceRecord(String line) { 256 return line.charAt(kIndex_CardDeviceField) == '['; 257 } 258 scan()259 public void scan() { 260 mDeviceRecords.clear(); 261 262 File devicesFile = new File(kDevicesFilePath); 263 try { 264 FileReader reader = new FileReader(devicesFile); 265 BufferedReader bufferedReader = new BufferedReader(reader); 266 String line = ""; 267 while ((line = bufferedReader.readLine()) != null) { 268 if (isLineDeviceRecord(line)) { 269 AlsaDeviceRecord deviceRecord = new AlsaDeviceRecord(); 270 deviceRecord.parse(line); 271 mDeviceRecords.add(deviceRecord); 272 } 273 } 274 reader.close(); 275 } catch (FileNotFoundException e) { 276 e.printStackTrace(); 277 } catch (IOException e) { 278 e.printStackTrace(); 279 } 280 } 281 282 // 283 // Loging 284 // Log(String heading)285 public void Log(String heading) { 286 if (DEBUG) { 287 Slog.i(TAG, heading); 288 for (AlsaDeviceRecord deviceRecord : mDeviceRecords) { 289 Slog.i(TAG, deviceRecord.textFormat()); 290 } 291 } 292 } 293 } // class AlsaDevicesParser 294 295