xonar_lib.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * helper functions for Asus Xonar cards
  4. *
  5. * Copyright (c) Clemens Ladisch <[email protected]>
  6. */
  7. #include <linux/delay.h>
  8. #include <sound/core.h>
  9. #include <sound/control.h>
  10. #include <sound/pcm.h>
  11. #include <sound/pcm_params.h>
  12. #include "xonar.h"
  13. #define GPIO_CS53x1_M_MASK 0x000c
  14. #define GPIO_CS53x1_M_SINGLE 0x0000
  15. #define GPIO_CS53x1_M_DOUBLE 0x0004
  16. #define GPIO_CS53x1_M_QUAD 0x0008
  17. void xonar_enable_output(struct oxygen *chip)
  18. {
  19. struct xonar_generic *data = chip->model_data;
  20. oxygen_set_bits16(chip, OXYGEN_GPIO_CONTROL, data->output_enable_bit);
  21. msleep(data->anti_pop_delay);
  22. oxygen_set_bits16(chip, OXYGEN_GPIO_DATA, data->output_enable_bit);
  23. }
  24. void xonar_disable_output(struct oxygen *chip)
  25. {
  26. struct xonar_generic *data = chip->model_data;
  27. oxygen_clear_bits16(chip, OXYGEN_GPIO_DATA, data->output_enable_bit);
  28. }
  29. static void xonar_ext_power_gpio_changed(struct oxygen *chip)
  30. {
  31. struct xonar_generic *data = chip->model_data;
  32. u8 has_power;
  33. has_power = !!(oxygen_read8(chip, data->ext_power_reg)
  34. & data->ext_power_bit);
  35. if (has_power != data->has_power) {
  36. data->has_power = has_power;
  37. if (has_power) {
  38. dev_notice(chip->card->dev, "power restored\n");
  39. } else {
  40. dev_crit(chip->card->dev,
  41. "Hey! Don't unplug the power cable!\n");
  42. /* TODO: stop PCMs */
  43. }
  44. }
  45. }
  46. void xonar_init_ext_power(struct oxygen *chip)
  47. {
  48. struct xonar_generic *data = chip->model_data;
  49. oxygen_set_bits8(chip, data->ext_power_int_reg,
  50. data->ext_power_bit);
  51. chip->interrupt_mask |= OXYGEN_INT_GPIO;
  52. chip->model.gpio_changed = xonar_ext_power_gpio_changed;
  53. data->has_power = !!(oxygen_read8(chip, data->ext_power_reg)
  54. & data->ext_power_bit);
  55. }
  56. void xonar_init_cs53x1(struct oxygen *chip)
  57. {
  58. oxygen_set_bits16(chip, OXYGEN_GPIO_CONTROL, GPIO_CS53x1_M_MASK);
  59. oxygen_write16_masked(chip, OXYGEN_GPIO_DATA,
  60. GPIO_CS53x1_M_SINGLE, GPIO_CS53x1_M_MASK);
  61. }
  62. void xonar_set_cs53x1_params(struct oxygen *chip,
  63. struct snd_pcm_hw_params *params)
  64. {
  65. unsigned int value;
  66. if (params_rate(params) <= 54000)
  67. value = GPIO_CS53x1_M_SINGLE;
  68. else if (params_rate(params) <= 108000)
  69. value = GPIO_CS53x1_M_DOUBLE;
  70. else
  71. value = GPIO_CS53x1_M_QUAD;
  72. oxygen_write16_masked(chip, OXYGEN_GPIO_DATA,
  73. value, GPIO_CS53x1_M_MASK);
  74. }
  75. int xonar_gpio_bit_switch_get(struct snd_kcontrol *ctl,
  76. struct snd_ctl_elem_value *value)
  77. {
  78. struct oxygen *chip = ctl->private_data;
  79. u16 bit = ctl->private_value;
  80. bool invert = ctl->private_value & XONAR_GPIO_BIT_INVERT;
  81. value->value.integer.value[0] =
  82. !!(oxygen_read16(chip, OXYGEN_GPIO_DATA) & bit) ^ invert;
  83. return 0;
  84. }
  85. int xonar_gpio_bit_switch_put(struct snd_kcontrol *ctl,
  86. struct snd_ctl_elem_value *value)
  87. {
  88. struct oxygen *chip = ctl->private_data;
  89. u16 bit = ctl->private_value;
  90. bool invert = ctl->private_value & XONAR_GPIO_BIT_INVERT;
  91. u16 old_bits, new_bits;
  92. int changed;
  93. spin_lock_irq(&chip->reg_lock);
  94. old_bits = oxygen_read16(chip, OXYGEN_GPIO_DATA);
  95. if (!!value->value.integer.value[0] ^ invert)
  96. new_bits = old_bits | bit;
  97. else
  98. new_bits = old_bits & ~bit;
  99. changed = new_bits != old_bits;
  100. if (changed)
  101. oxygen_write16(chip, OXYGEN_GPIO_DATA, new_bits);
  102. spin_unlock_irq(&chip->reg_lock);
  103. return changed;
  104. }