aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/video
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/media/video')
-rw-r--r--drivers/media/video/cs53l32a.c65
1 files changed, 37 insertions, 28 deletions
diff --git a/drivers/media/video/cs53l32a.c b/drivers/media/video/cs53l32a.c
index aa31b6736a7a..7d997aeda63e 100644
--- a/drivers/media/video/cs53l32a.c
+++ b/drivers/media/video/cs53l32a.c
@@ -74,50 +74,59 @@ static int cs53l32a_read(struct i2c_client *client, u8 reg)
74static int cs53l32a_command(struct i2c_client *client, unsigned int cmd, 74static int cs53l32a_command(struct i2c_client *client, unsigned int cmd,
75 void *arg) 75 void *arg)
76{ 76{
77 int *input = arg; 77 struct v4l2_audio *input = arg;
78 struct v4l2_control *ctrl = arg;
78 79
79 switch (cmd) { 80 switch (cmd) {
80 case AUDC_SET_INPUT: 81 case VIDIOC_S_AUDIO:
81 switch (*input) { 82 /* There are 2 physical inputs, but the second input can be
82 case AUDIO_TUNER: 83 placed in two modes, the first mode bypasses the PGA (gain),
83 cs53l32a_write(client, 0x01, 0x01); 84 the second goes through the PGA. Hence there are three
84 break; 85 possible inputs to choose from. */
85 case AUDIO_EXTERN: 86 if (input->index > 2) {
86 cs53l32a_write(client, 0x01, 0x21); 87 cs53l32a_err("Invalid input %d.\n", input->index);
87 break;
88 case AUDIO_MUTE:
89 cs53l32a_write(client, 0x03, 0xF0);
90 break;
91 case AUDIO_UNMUTE:
92 cs53l32a_write(client, 0x03, 0x30);
93 break;
94 default:
95 cs53l32a_err("Invalid input %d.\n", *input);
96 return -EINVAL; 88 return -EINVAL;
97 } 89 }
90 cs53l32a_write(client, 0x01, 0x01 + (input->index << 4));
91 break;
92
93 case VIDIOC_G_AUDIO:
94 memset(input, 0, sizeof(*input));
95 input->index = (cs53l32a_read(client, 0x01) >> 4) & 3;
96 break;
97
98 case VIDIOC_G_CTRL:
99 if (ctrl->id == V4L2_CID_AUDIO_MUTE) {
100 ctrl->value = (cs53l32a_read(client, 0x03) & 0xc0) != 0;
101 break;
102 }
103 if (ctrl->id != V4L2_CID_AUDIO_VOLUME)
104 return -EINVAL;
105 ctrl->value = (s8)cs53l32a_read(client, 0x04);
98 break; 106 break;
99 107
100 case VIDIOC_S_CTRL: 108 case VIDIOC_S_CTRL:
101 { 109 if (ctrl->id == V4L2_CID_AUDIO_MUTE) {
102 struct v4l2_control *ctrl = arg; 110 cs53l32a_write(client, 0x03, ctrl->value ? 0xf0 : 0x30);
103
104 if (ctrl->id != V4L2_CID_AUDIO_VOLUME)
105 return -EINVAL;
106 if (ctrl->value > 12 || ctrl->value < -90)
107 return -EINVAL;
108 cs53l32a_write(client, 0x04, (u8) ctrl->value);
109 cs53l32a_write(client, 0x05, (u8) ctrl->value);
110 break; 111 break;
111 } 112 }
113 if (ctrl->id != V4L2_CID_AUDIO_VOLUME)
114 return -EINVAL;
115 if (ctrl->value > 12 || ctrl->value < -96)
116 return -EINVAL;
117 cs53l32a_write(client, 0x04, (u8) ctrl->value);
118 cs53l32a_write(client, 0x05, (u8) ctrl->value);
119 break;
112 120
113 case VIDIOC_LOG_STATUS: 121 case VIDIOC_LOG_STATUS:
114 { 122 {
115 u8 v = cs53l32a_read(client, 0x01); 123 u8 v = cs53l32a_read(client, 0x01);
116 u8 m = cs53l32a_read(client, 0x03); 124 u8 m = cs53l32a_read(client, 0x03);
125 s8 vol = cs53l32a_read(client, 0x04);
117 126
118 cs53l32a_info("Input: %s%s\n", 127 cs53l32a_info("Input: %d%s\n", (v >> 4) & 3,
119 v == 0x21 ? "external line in" : "tuner",
120 (m & 0xC0) ? " (muted)" : ""); 128 (m & 0xC0) ? " (muted)" : "");
129 cs53l32a_info("Volume: %d dB\n", vol);
121 break; 130 break;
122 } 131 }
123 132