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 17 package android.media.audiopolicy; 18 19 import android.annotation.SystemApi; 20 import android.media.AudioAttributes; 21 import android.os.Parcel; 22 import android.util.Log; 23 24 import java.util.ArrayList; 25 import java.util.Iterator; 26 import java.util.Objects; 27 28 29 /** 30 * @hide 31 * 32 * Here's an example of creating a mixing rule for all media playback: 33 * <pre> 34 * AudioAttributes mediaAttr = new AudioAttributes.Builder() 35 * .setUsage(AudioAttributes.USAGE_MEDIA) 36 * .build(); 37 * AudioMixingRule mediaRule = new AudioMixingRule.Builder() 38 * .addRule(mediaAttr, AudioMixingRule.RULE_MATCH_ATTRIBUTE_USAGE) 39 * .build(); 40 * </pre> 41 */ 42 @SystemApi 43 public class AudioMixingRule { 44 AudioMixingRule(int mixType, ArrayList<AudioMixMatchCriterion> criteria)45 private AudioMixingRule(int mixType, ArrayList<AudioMixMatchCriterion> criteria) { 46 mCriteria = criteria; 47 mTargetMixType = mixType; 48 } 49 50 /** 51 * A rule requiring the usage information of the {@link AudioAttributes} to match. 52 * This mixing rule can be added with {@link Builder#addRule(AudioAttributes, int)} or 53 * {@link Builder#addMixRule(int, Object)} where the Object parameter is an instance of 54 * {@link AudioAttributes}. 55 */ 56 @SystemApi 57 public static final int RULE_MATCH_ATTRIBUTE_USAGE = 0x1; 58 /** 59 * A rule requiring the capture preset information of the {@link AudioAttributes} to match. 60 * This mixing rule can be added with {@link Builder#addRule(AudioAttributes, int)} or 61 * {@link Builder#addMixRule(int, Object)} where the Object parameter is an instance of 62 * {@link AudioAttributes}. 63 */ 64 @SystemApi 65 public static final int RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET = 0x1 << 1; 66 /** 67 * A rule requiring the UID of the audio stream to match that specified. 68 * This mixing rule can be added with {@link Builder#addMixRule(int, Object)} where the Object 69 * parameter is an instance of {@link java.lang.Integer}. 70 */ 71 @SystemApi 72 public static final int RULE_MATCH_UID = 0x1 << 2; 73 74 private final static int RULE_EXCLUSION_MASK = 0x8000; 75 /** 76 * @hide 77 * A rule requiring the usage information of the {@link AudioAttributes} to differ. 78 */ 79 public static final int RULE_EXCLUDE_ATTRIBUTE_USAGE = 80 RULE_EXCLUSION_MASK | RULE_MATCH_ATTRIBUTE_USAGE; 81 /** 82 * @hide 83 * A rule requiring the capture preset information of the {@link AudioAttributes} to differ. 84 */ 85 public static final int RULE_EXCLUDE_ATTRIBUTE_CAPTURE_PRESET = 86 RULE_EXCLUSION_MASK | RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET; 87 /** 88 * @hide 89 * A rule requiring the UID information to differ. 90 */ 91 public static final int RULE_EXCLUDE_UID = 92 RULE_EXCLUSION_MASK | RULE_MATCH_UID; 93 94 static final class AudioMixMatchCriterion { 95 final AudioAttributes mAttr; 96 final int mIntProp; 97 final int mRule; 98 99 /** input parameters must be valid */ AudioMixMatchCriterion(AudioAttributes attributes, int rule)100 AudioMixMatchCriterion(AudioAttributes attributes, int rule) { 101 mAttr = attributes; 102 mIntProp = Integer.MIN_VALUE; 103 mRule = rule; 104 } 105 /** input parameters must be valid */ AudioMixMatchCriterion(Integer intProp, int rule)106 AudioMixMatchCriterion(Integer intProp, int rule) { 107 mAttr = null; 108 mIntProp = intProp.intValue(); 109 mRule = rule; 110 } 111 112 @Override hashCode()113 public int hashCode() { 114 return Objects.hash(mAttr, mIntProp, mRule); 115 } 116 writeToParcel(Parcel dest)117 void writeToParcel(Parcel dest) { 118 dest.writeInt(mRule); 119 final int match_rule = mRule & ~RULE_EXCLUSION_MASK; 120 switch (match_rule) { 121 case RULE_MATCH_ATTRIBUTE_USAGE: 122 dest.writeInt(mAttr.getUsage()); 123 break; 124 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 125 dest.writeInt(mAttr.getCapturePreset()); 126 break; 127 case RULE_MATCH_UID: 128 dest.writeInt(mIntProp); 129 break; 130 default: 131 Log.e("AudioMixMatchCriterion", "Unknown match rule" + match_rule 132 + " when writing to Parcel"); 133 dest.writeInt(-1); 134 } 135 } 136 } 137 138 private final int mTargetMixType; getTargetMixType()139 int getTargetMixType() { return mTargetMixType; } 140 private final ArrayList<AudioMixMatchCriterion> mCriteria; getCriteria()141 ArrayList<AudioMixMatchCriterion> getCriteria() { return mCriteria; } 142 143 @Override hashCode()144 public int hashCode() { 145 return Objects.hash(mTargetMixType, mCriteria); 146 } 147 isValidSystemApiRule(int rule)148 private static boolean isValidSystemApiRule(int rule) { 149 // API rules only expose the RULE_MATCH_* rules 150 switch (rule) { 151 case RULE_MATCH_ATTRIBUTE_USAGE: 152 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 153 case RULE_MATCH_UID: 154 return true; 155 default: 156 return false; 157 } 158 } isValidAttributesSystemApiRule(int rule)159 private static boolean isValidAttributesSystemApiRule(int rule) { 160 // API rules only expose the RULE_MATCH_* rules 161 switch (rule) { 162 case RULE_MATCH_ATTRIBUTE_USAGE: 163 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 164 return true; 165 default: 166 return false; 167 } 168 } 169 isValidRule(int rule)170 private static boolean isValidRule(int rule) { 171 final int match_rule = rule & ~RULE_EXCLUSION_MASK; 172 switch (match_rule) { 173 case RULE_MATCH_ATTRIBUTE_USAGE: 174 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 175 case RULE_MATCH_UID: 176 return true; 177 default: 178 return false; 179 } 180 } 181 isPlayerRule(int rule)182 private static boolean isPlayerRule(int rule) { 183 final int match_rule = rule & ~RULE_EXCLUSION_MASK; 184 switch (match_rule) { 185 case RULE_MATCH_ATTRIBUTE_USAGE: 186 case RULE_MATCH_UID: 187 return true; 188 default: 189 return false; 190 } 191 } 192 isAudioAttributeRule(int match_rule)193 private static boolean isAudioAttributeRule(int match_rule) { 194 switch(match_rule) { 195 case RULE_MATCH_ATTRIBUTE_USAGE: 196 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 197 return true; 198 default: 199 return false; 200 } 201 } 202 203 /** 204 * Builder class for {@link AudioMixingRule} objects 205 */ 206 @SystemApi 207 public static class Builder { 208 private ArrayList<AudioMixMatchCriterion> mCriteria; 209 private int mTargetMixType = AudioMix.MIX_TYPE_INVALID; 210 211 /** 212 * Constructs a new Builder with no rules. 213 */ 214 @SystemApi Builder()215 public Builder() { 216 mCriteria = new ArrayList<AudioMixMatchCriterion>(); 217 } 218 219 /** 220 * Add a rule for the selection of which streams are mixed together. 221 * @param attrToMatch a non-null AudioAttributes instance for which a contradictory 222 * rule hasn't been set yet. 223 * @param rule {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_USAGE} or 224 * {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET}. 225 * @return the same Builder instance. 226 * @throws IllegalArgumentException 227 * @see #excludeRule(AudioAttributes, int) 228 */ 229 @SystemApi addRule(AudioAttributes attrToMatch, int rule)230 public Builder addRule(AudioAttributes attrToMatch, int rule) 231 throws IllegalArgumentException { 232 if (!isValidAttributesSystemApiRule(rule)) { 233 throw new IllegalArgumentException("Illegal rule value " + rule); 234 } 235 return checkAddRuleObjInternal(rule, attrToMatch); 236 } 237 238 /** 239 * Add a rule by exclusion for the selection of which streams are mixed together. 240 * <br>For instance the following code 241 * <br><pre> 242 * AudioAttributes mediaAttr = new AudioAttributes.Builder() 243 * .setUsage(AudioAttributes.USAGE_MEDIA) 244 * .build(); 245 * AudioMixingRule noMediaRule = new AudioMixingRule.Builder() 246 * .excludeRule(mediaAttr, AudioMixingRule.RULE_MATCH_ATTRIBUTE_USAGE) 247 * .build(); 248 * </pre> 249 * <br>will create a rule which maps to any usage value, except USAGE_MEDIA. 250 * @param attrToMatch a non-null AudioAttributes instance for which a contradictory 251 * rule hasn't been set yet. 252 * @param rule {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_USAGE} or 253 * {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET}. 254 * @return the same Builder instance. 255 * @throws IllegalArgumentException 256 * @see #addRule(AudioAttributes, int) 257 */ 258 @SystemApi excludeRule(AudioAttributes attrToMatch, int rule)259 public Builder excludeRule(AudioAttributes attrToMatch, int rule) 260 throws IllegalArgumentException { 261 if (!isValidAttributesSystemApiRule(rule)) { 262 throw new IllegalArgumentException("Illegal rule value " + rule); 263 } 264 return checkAddRuleObjInternal(rule | RULE_EXCLUSION_MASK, attrToMatch); 265 } 266 267 /** 268 * Add a rule for the selection of which streams are mixed together. 269 * The rule defines what the matching will be made on. It also determines the type of the 270 * property to match against. 271 * @param rule one of {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_USAGE}, 272 * {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET} or 273 * {@link AudioMixingRule#RULE_MATCH_UID}. 274 * @param property see the definition of each rule for the type to use (either an 275 * {@link AudioAttributes} or an {@link java.lang.Integer}). 276 * @return the same Builder instance. 277 * @throws IllegalArgumentException 278 * @see #excludeMixRule(int, Object) 279 */ 280 @SystemApi addMixRule(int rule, Object property)281 public Builder addMixRule(int rule, Object property) throws IllegalArgumentException { 282 if (!isValidSystemApiRule(rule)) { 283 throw new IllegalArgumentException("Illegal rule value " + rule); 284 } 285 return checkAddRuleObjInternal(rule, property); 286 } 287 288 /** 289 * Add a rule by exclusion for the selection of which streams are mixed together. 290 * <br>For instance the following code 291 * <br><pre> 292 * AudioAttributes mediaAttr = new AudioAttributes.Builder() 293 * .setUsage(AudioAttributes.USAGE_MEDIA) 294 * .build(); 295 * AudioMixingRule noMediaRule = new AudioMixingRule.Builder() 296 * .addMixRule(AudioMixingRule.RULE_MATCH_ATTRIBUTE_USAGE, mediaAttr) 297 * .excludeMixRule(AudioMixingRule.RULE_MATCH_UID, new Integer(uidToExclude) 298 * .build(); 299 * </pre> 300 * <br>will create a rule which maps to usage USAGE_MEDIA, but excludes any stream 301 * coming from the specified UID. 302 * @param rule one of {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_USAGE}, 303 * {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET} or 304 * {@link AudioMixingRule#RULE_MATCH_UID}. 305 * @param property see the definition of each rule for the type to use (either an 306 * {@link AudioAttributes} or an {@link java.lang.Integer}). 307 * @return the same Builder instance. 308 * @throws IllegalArgumentException 309 */ 310 @SystemApi excludeMixRule(int rule, Object property)311 public Builder excludeMixRule(int rule, Object property) throws IllegalArgumentException { 312 if (!isValidSystemApiRule(rule)) { 313 throw new IllegalArgumentException("Illegal rule value " + rule); 314 } 315 return checkAddRuleObjInternal(rule | RULE_EXCLUSION_MASK, property); 316 } 317 318 /** 319 * Add or exclude a rule for the selection of which streams are mixed together. 320 * Does error checking on the parameters. 321 * @param rule 322 * @param property 323 * @return the same Builder instance. 324 * @throws IllegalArgumentException 325 */ checkAddRuleObjInternal(int rule, Object property)326 private Builder checkAddRuleObjInternal(int rule, Object property) 327 throws IllegalArgumentException { 328 if (property == null) { 329 throw new IllegalArgumentException("Illegal null argument for mixing rule"); 330 } 331 if (!isValidRule(rule)) { 332 throw new IllegalArgumentException("Illegal rule value " + rule); 333 } 334 final int match_rule = rule & ~RULE_EXCLUSION_MASK; 335 if (isAudioAttributeRule(match_rule)) { 336 if (!(property instanceof AudioAttributes)) { 337 throw new IllegalArgumentException("Invalid AudioAttributes argument"); 338 } 339 return addRuleInternal((AudioAttributes) property, null, rule); 340 } else { 341 // implies integer match rule 342 if (!(property instanceof Integer)) { 343 throw new IllegalArgumentException("Invalid Integer argument"); 344 } 345 return addRuleInternal(null, (Integer) property, rule); 346 } 347 } 348 349 /** 350 * Add or exclude a rule on AudioAttributes or integer property for the selection of which 351 * streams are mixed together. 352 * No rule-to-parameter type check, all done in {@link #checkAddRuleObjInternal(int, Object)}. 353 * Exceptions are thrown only when incompatible rules are added. 354 * @param attrToMatch a non-null AudioAttributes instance for which a contradictory 355 * rule hasn't been set yet, null if not used. 356 * @param intProp an integer property to match or exclude, null if not used. 357 * @param rule one of {@link AudioMixingRule#RULE_EXCLUDE_ATTRIBUTE_USAGE}, 358 * {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_USAGE}, 359 * {@link AudioMixingRule#RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET} or 360 * {@link AudioMixingRule#RULE_EXCLUDE_ATTRIBUTE_CAPTURE_PRESET}, 361 * {@link AudioMixingRule#RULE_MATCH_UID}, {@link AudioMixingRule#RULE_EXCLUDE_UID}. 362 * @return the same Builder instance. 363 * @throws IllegalArgumentException 364 */ addRuleInternal(AudioAttributes attrToMatch, Integer intProp, int rule)365 private Builder addRuleInternal(AudioAttributes attrToMatch, Integer intProp, int rule) 366 throws IllegalArgumentException { 367 // as rules are added to the Builder, we verify they are consistent with the type 368 // of mix being built. When adding the first rule, the mix type is MIX_TYPE_INVALID. 369 if (mTargetMixType == AudioMix.MIX_TYPE_INVALID) { 370 if (isPlayerRule(rule)) { 371 mTargetMixType = AudioMix.MIX_TYPE_PLAYERS; 372 } else { 373 mTargetMixType = AudioMix.MIX_TYPE_RECORDERS; 374 } 375 } else if (((mTargetMixType == AudioMix.MIX_TYPE_PLAYERS) && !isPlayerRule(rule)) 376 || ((mTargetMixType == AudioMix.MIX_TYPE_RECORDERS) && isPlayerRule(rule))) 377 { 378 throw new IllegalArgumentException("Incompatible rule for mix"); 379 } 380 synchronized (mCriteria) { 381 Iterator<AudioMixMatchCriterion> crIterator = mCriteria.iterator(); 382 final int match_rule = rule & ~RULE_EXCLUSION_MASK; 383 while (crIterator.hasNext()) { 384 final AudioMixMatchCriterion criterion = crIterator.next(); 385 switch (match_rule) { 386 case RULE_MATCH_ATTRIBUTE_USAGE: 387 // "usage"-based rule 388 if (criterion.mAttr.getUsage() == attrToMatch.getUsage()) { 389 if (criterion.mRule == rule) { 390 // rule already exists, we're done 391 return this; 392 } else { 393 // criterion already exists with a another rule, 394 // it is incompatible 395 throw new IllegalArgumentException("Contradictory rule exists" 396 + " for " + attrToMatch); 397 } 398 } 399 break; 400 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 401 // "capture preset"-base rule 402 if (criterion.mAttr.getCapturePreset() == attrToMatch.getCapturePreset()) { 403 if (criterion.mRule == rule) { 404 // rule already exists, we're done 405 return this; 406 } else { 407 // criterion already exists with a another rule, 408 // it is incompatible 409 throw new IllegalArgumentException("Contradictory rule exists" 410 + " for " + attrToMatch); 411 } 412 } 413 break; 414 case RULE_MATCH_UID: 415 // "usage"-based rule 416 if (criterion.mIntProp == intProp.intValue()) { 417 if (criterion.mRule == rule) { 418 // rule already exists, we're done 419 return this; 420 } else { 421 // criterion already exists with a another rule, 422 // it is incompatible 423 throw new IllegalArgumentException("Contradictory rule exists" 424 + " for UID " + intProp); 425 } 426 } 427 break; 428 } 429 } 430 // rule didn't exist, add it 431 switch (match_rule) { 432 case RULE_MATCH_ATTRIBUTE_USAGE: 433 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 434 mCriteria.add(new AudioMixMatchCriterion(attrToMatch, rule)); 435 break; 436 case RULE_MATCH_UID: 437 mCriteria.add(new AudioMixMatchCriterion(intProp, rule)); 438 break; 439 default: 440 throw new IllegalStateException("Unreachable code in addRuleInternal()"); 441 } 442 } 443 return this; 444 } 445 addRuleFromParcel(Parcel in)446 Builder addRuleFromParcel(Parcel in) throws IllegalArgumentException { 447 final int rule = in.readInt(); 448 final int match_rule = rule & ~RULE_EXCLUSION_MASK; 449 AudioAttributes attr = null; 450 Integer intProp = null; 451 switch (match_rule) { 452 case RULE_MATCH_ATTRIBUTE_USAGE: 453 int usage = in.readInt(); 454 attr = new AudioAttributes.Builder() 455 .setUsage(usage).build(); 456 break; 457 case RULE_MATCH_ATTRIBUTE_CAPTURE_PRESET: 458 int preset = in.readInt(); 459 attr = new AudioAttributes.Builder() 460 .setInternalCapturePreset(preset).build(); 461 break; 462 case RULE_MATCH_UID: 463 intProp = new Integer(in.readInt()); 464 break; 465 default: 466 // assume there was in int value to read as for now they come in pair 467 in.readInt(); 468 throw new IllegalArgumentException("Illegal rule value " + rule + " in parcel"); 469 } 470 return addRuleInternal(attr, intProp, rule); 471 } 472 473 /** 474 * Combines all of the matching and exclusion rules that have been set and return a new 475 * {@link AudioMixingRule} object. 476 * @return a new {@link AudioMixingRule} object 477 */ build()478 public AudioMixingRule build() { 479 return new AudioMixingRule(mTargetMixType, mCriteria); 480 } 481 } 482 } 483