1/*
2 * drivers/mfd/si476x-cmd.c -- Subroutines implementing command
3 * protocol of si476x series of chips
4 *
5 * Copyright (C) 2012 Innovative Converged Devices(ICD)
6 * Copyright (C) 2013 Andrey Smirnov
7 *
8 * Author: Andrey Smirnov <andrew.smirnov@gmail.com>
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; version 2 of the License.
13 *
14 * This program is distributed in the hope that it will be useful, but
15 * WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17 * General Public License for more details.
18 *
19 */
20
21#include <linux/module.h>
22#include <linux/completion.h>
23#include <linux/delay.h>
24#include <linux/atomic.h>
25#include <linux/i2c.h>
26#include <linux/device.h>
27#include <linux/gpio.h>
28#include <linux/videodev2.h>
29
30#include <linux/mfd/si476x-core.h>
31
32#include <asm/unaligned.h>
33
34#define msb(x)                  ((u8)((u16) x >> 8))
35#define lsb(x)                  ((u8)((u16) x &  0x00FF))
36
37
38
39#define CMD_POWER_UP				0x01
40#define CMD_POWER_UP_A10_NRESP			1
41#define CMD_POWER_UP_A10_NARGS			5
42
43#define CMD_POWER_UP_A20_NRESP			1
44#define CMD_POWER_UP_A20_NARGS			5
45
46#define POWER_UP_DELAY_MS			110
47
48#define CMD_POWER_DOWN				0x11
49#define CMD_POWER_DOWN_A10_NRESP		1
50
51#define CMD_POWER_DOWN_A20_NRESP		1
52#define CMD_POWER_DOWN_A20_NARGS		1
53
54#define CMD_FUNC_INFO				0x12
55#define CMD_FUNC_INFO_NRESP			7
56
57#define CMD_SET_PROPERTY			0x13
58#define CMD_SET_PROPERTY_NARGS			5
59#define CMD_SET_PROPERTY_NRESP			1
60
61#define CMD_GET_PROPERTY			0x14
62#define CMD_GET_PROPERTY_NARGS			3
63#define CMD_GET_PROPERTY_NRESP			4
64
65#define CMD_AGC_STATUS				0x17
66#define CMD_AGC_STATUS_NRESP_A10		2
67#define CMD_AGC_STATUS_NRESP_A20                6
68
69#define PIN_CFG_BYTE(x) (0x7F & (x))
70#define CMD_DIG_AUDIO_PIN_CFG			0x18
71#define CMD_DIG_AUDIO_PIN_CFG_NARGS		4
72#define CMD_DIG_AUDIO_PIN_CFG_NRESP		5
73
74#define CMD_ZIF_PIN_CFG				0x19
75#define CMD_ZIF_PIN_CFG_NARGS			4
76#define CMD_ZIF_PIN_CFG_NRESP			5
77
78#define CMD_IC_LINK_GPO_CTL_PIN_CFG		0x1A
79#define CMD_IC_LINK_GPO_CTL_PIN_CFG_NARGS	4
80#define CMD_IC_LINK_GPO_CTL_PIN_CFG_NRESP	5
81
82#define CMD_ANA_AUDIO_PIN_CFG			0x1B
83#define CMD_ANA_AUDIO_PIN_CFG_NARGS		1
84#define CMD_ANA_AUDIO_PIN_CFG_NRESP		2
85
86#define CMD_INTB_PIN_CFG			0x1C
87#define CMD_INTB_PIN_CFG_NARGS			2
88#define CMD_INTB_PIN_CFG_A10_NRESP		6
89#define CMD_INTB_PIN_CFG_A20_NRESP		3
90
91#define CMD_FM_TUNE_FREQ			0x30
92#define CMD_FM_TUNE_FREQ_A10_NARGS		5
93#define CMD_FM_TUNE_FREQ_A20_NARGS		3
94#define CMD_FM_TUNE_FREQ_NRESP			1
95
96#define CMD_FM_RSQ_STATUS			0x32
97
98#define CMD_FM_RSQ_STATUS_A10_NARGS		1
99#define CMD_FM_RSQ_STATUS_A10_NRESP		17
100#define CMD_FM_RSQ_STATUS_A30_NARGS		1
101#define CMD_FM_RSQ_STATUS_A30_NRESP		23
102
103
104#define CMD_FM_SEEK_START			0x31
105#define CMD_FM_SEEK_START_NARGS			1
106#define CMD_FM_SEEK_START_NRESP			1
107
108#define CMD_FM_RDS_STATUS			0x36
109#define CMD_FM_RDS_STATUS_NARGS			1
110#define CMD_FM_RDS_STATUS_NRESP			16
111
112#define CMD_FM_RDS_BLOCKCOUNT			0x37
113#define CMD_FM_RDS_BLOCKCOUNT_NARGS		1
114#define CMD_FM_RDS_BLOCKCOUNT_NRESP		8
115
116#define CMD_FM_PHASE_DIVERSITY			0x38
117#define CMD_FM_PHASE_DIVERSITY_NARGS		1
118#define CMD_FM_PHASE_DIVERSITY_NRESP		1
119
120#define CMD_FM_PHASE_DIV_STATUS			0x39
121#define CMD_FM_PHASE_DIV_STATUS_NRESP		2
122
123#define CMD_AM_TUNE_FREQ			0x40
124#define CMD_AM_TUNE_FREQ_NARGS			3
125#define CMD_AM_TUNE_FREQ_NRESP			1
126
127#define CMD_AM_RSQ_STATUS			0x42
128#define CMD_AM_RSQ_STATUS_NARGS			1
129#define CMD_AM_RSQ_STATUS_NRESP			13
130
131#define CMD_AM_SEEK_START			0x41
132#define CMD_AM_SEEK_START_NARGS			1
133#define CMD_AM_SEEK_START_NRESP			1
134
135
136#define CMD_AM_ACF_STATUS			0x45
137#define CMD_AM_ACF_STATUS_NRESP			6
138#define CMD_AM_ACF_STATUS_NARGS			1
139
140#define CMD_FM_ACF_STATUS			0x35
141#define CMD_FM_ACF_STATUS_NRESP			8
142#define CMD_FM_ACF_STATUS_NARGS			1
143
144#define CMD_MAX_ARGS_COUNT			(10)
145
146
147enum si476x_acf_status_report_bits {
148	SI476X_ACF_BLEND_INT	= (1 << 4),
149	SI476X_ACF_HIBLEND_INT	= (1 << 3),
150	SI476X_ACF_HICUT_INT	= (1 << 2),
151	SI476X_ACF_CHBW_INT	= (1 << 1),
152	SI476X_ACF_SOFTMUTE_INT	= (1 << 0),
153
154	SI476X_ACF_SMUTE	= (1 << 0),
155	SI476X_ACF_SMATTN	= 0x1f,
156	SI476X_ACF_PILOT	= (1 << 7),
157	SI476X_ACF_STBLEND	= ~SI476X_ACF_PILOT,
158};
159
160enum si476x_agc_status_report_bits {
161	SI476X_AGC_MXHI		= (1 << 5),
162	SI476X_AGC_MXLO		= (1 << 4),
163	SI476X_AGC_LNAHI	= (1 << 3),
164	SI476X_AGC_LNALO	= (1 << 2),
165};
166
167enum si476x_errors {
168	SI476X_ERR_BAD_COMMAND		= 0x10,
169	SI476X_ERR_BAD_ARG1		= 0x11,
170	SI476X_ERR_BAD_ARG2		= 0x12,
171	SI476X_ERR_BAD_ARG3		= 0x13,
172	SI476X_ERR_BAD_ARG4		= 0x14,
173	SI476X_ERR_BUSY			= 0x18,
174	SI476X_ERR_BAD_INTERNAL_MEMORY  = 0x20,
175	SI476X_ERR_BAD_PATCH		= 0x30,
176	SI476X_ERR_BAD_BOOT_MODE	= 0x31,
177	SI476X_ERR_BAD_PROPERTY		= 0x40,
178};
179
180static int si476x_core_parse_and_nag_about_error(struct si476x_core *core)
181{
182	int err;
183	char *cause;
184	u8 buffer[2];
185
186	if (core->revision != SI476X_REVISION_A10) {
187		err = si476x_core_i2c_xfer(core, SI476X_I2C_RECV,
188					   buffer, sizeof(buffer));
189		if (err == sizeof(buffer)) {
190			switch (buffer[1]) {
191			case SI476X_ERR_BAD_COMMAND:
192				cause = "Bad command";
193				err = -EINVAL;
194				break;
195			case SI476X_ERR_BAD_ARG1:
196				cause = "Bad argument #1";
197				err = -EINVAL;
198				break;
199			case SI476X_ERR_BAD_ARG2:
200				cause = "Bad argument #2";
201				err = -EINVAL;
202				break;
203			case SI476X_ERR_BAD_ARG3:
204				cause = "Bad argument #3";
205				err = -EINVAL;
206				break;
207			case SI476X_ERR_BAD_ARG4:
208				cause = "Bad argument #4";
209				err = -EINVAL;
210				break;
211			case SI476X_ERR_BUSY:
212				cause = "Chip is busy";
213				err = -EBUSY;
214				break;
215			case SI476X_ERR_BAD_INTERNAL_MEMORY:
216				cause = "Bad internal memory";
217				err = -EIO;
218				break;
219			case SI476X_ERR_BAD_PATCH:
220				cause = "Bad patch";
221				err = -EINVAL;
222				break;
223			case SI476X_ERR_BAD_BOOT_MODE:
224				cause = "Bad boot mode";
225				err = -EINVAL;
226				break;
227			case SI476X_ERR_BAD_PROPERTY:
228				cause = "Bad property";
229				err = -EINVAL;
230				break;
231			default:
232				cause = "Unknown";
233				err = -EIO;
234			}
235
236			dev_err(&core->client->dev,
237				"[Chip error status]: %s\n", cause);
238		} else {
239			dev_err(&core->client->dev,
240				"Failed to fetch error code\n");
241			err = (err >= 0) ? -EIO : err;
242		}
243	} else {
244		err = -EIO;
245	}
246
247	return err;
248}
249
250/**
251 * si476x_core_send_command() - sends a command to si476x and waits its
252 * response
253 * @core:    si476x_device structure for the device we are
254 *            communicating with
255 * @command:  command id
256 * @args:     command arguments we are sending
257 * @argn:     actual size of @args
258 * @response: buffer to place the expected response from the device
259 * @respn:    actual size of @response
260 * @usecs:    amount of time to wait before reading the response (in
261 *            usecs)
262 *
263 * Function returns 0 on succsess and negative error code on
264 * failure
265 */
266static int si476x_core_send_command(struct si476x_core *core,
267				    const u8 command,
268				    const u8 args[],
269				    const int argn,
270				    u8 resp[],
271				    const int respn,
272				    const int usecs)
273{
274	struct i2c_client *client = core->client;
275	int err;
276	u8  data[CMD_MAX_ARGS_COUNT + 1];
277
278	if (argn > CMD_MAX_ARGS_COUNT) {
279		err = -ENOMEM;
280		goto exit;
281	}
282
283	if (!client->adapter) {
284		err = -ENODEV;
285		goto exit;
286	}
287
288	/* First send the command and its arguments */
289	data[0] = command;
290	memcpy(&data[1], args, argn);
291	dev_dbg(&client->dev, "Command:\n %*ph\n", argn + 1, data);
292
293	err = si476x_core_i2c_xfer(core, SI476X_I2C_SEND,
294				   (char *) data, argn + 1);
295	if (err != argn + 1) {
296		dev_err(&core->client->dev,
297			"Error while sending command 0x%02x\n",
298			command);
299		err = (err >= 0) ? -EIO : err;
300		goto exit;
301	}
302	/* Set CTS to zero only after the command is send to avoid
303	 * possible racing conditions when working in polling mode */
304	atomic_set(&core->cts, 0);
305
306	/* if (unlikely(command == CMD_POWER_DOWN) */
307	if (!wait_event_timeout(core->command,
308				atomic_read(&core->cts),
309				usecs_to_jiffies(usecs) + 1))
310		dev_warn(&core->client->dev,
311			 "(%s) [CMD 0x%02x] Answer timeout.\n",
312			 __func__, command);
313
314	/*
315	  When working in polling mode, for some reason the tuner will
316	  report CTS bit as being set in the first status byte read,
317	  but all the consequtive ones will return zeros until the
318	  tuner is actually completed the POWER_UP command. To
319	  workaround that we wait for second CTS to be reported
320	 */
321	if (unlikely(!core->client->irq && command == CMD_POWER_UP)) {
322		if (!wait_event_timeout(core->command,
323					atomic_read(&core->cts),
324					usecs_to_jiffies(usecs) + 1))
325			dev_warn(&core->client->dev,
326				 "(%s) Power up took too much time.\n",
327				 __func__);
328	}
329
330	/* Then get the response */
331	err = si476x_core_i2c_xfer(core, SI476X_I2C_RECV, resp, respn);
332	if (err != respn) {
333		dev_err(&core->client->dev,
334			"Error while reading response for command 0x%02x\n",
335			command);
336		err = (err >= 0) ? -EIO : err;
337		goto exit;
338	}
339	dev_dbg(&client->dev, "Response:\n %*ph\n", respn, resp);
340
341	err = 0;
342
343	if (resp[0] & SI476X_ERR) {
344		dev_err(&core->client->dev,
345			"[CMD 0x%02x] Chip set error flag\n", command);
346		err = si476x_core_parse_and_nag_about_error(core);
347		goto exit;
348	}
349
350	if (!(resp[0] & SI476X_CTS))
351		err = -EBUSY;
352exit:
353	return err;
354}
355
356static int si476x_cmd_clear_stc(struct si476x_core *core)
357{
358	int err;
359	struct si476x_rsq_status_args args = {
360		.primary	= false,
361		.rsqack		= false,
362		.attune		= false,
363		.cancel		= false,
364		.stcack		= true,
365	};
366
367	switch (core->power_up_parameters.func) {
368	case SI476X_FUNC_FM_RECEIVER:
369		err = si476x_core_cmd_fm_rsq_status(core, &args, NULL);
370		break;
371	case SI476X_FUNC_AM_RECEIVER:
372		err = si476x_core_cmd_am_rsq_status(core, &args, NULL);
373		break;
374	default:
375		err = -EINVAL;
376	}
377
378	return err;
379}
380
381static int si476x_cmd_tune_seek_freq(struct si476x_core *core,
382				     uint8_t cmd,
383				     const uint8_t args[], size_t argn,
384				     uint8_t *resp, size_t respn)
385{
386	int err;
387
388
389	atomic_set(&core->stc, 0);
390	err = si476x_core_send_command(core, cmd, args, argn, resp, respn,
391				       SI476X_TIMEOUT_TUNE);
392	if (!err) {
393		wait_event_killable(core->tuning,
394				    atomic_read(&core->stc));
395		si476x_cmd_clear_stc(core);
396	}
397
398	return err;
399}
400
401/**
402 * si476x_cmd_func_info() - send 'FUNC_INFO' command to the device
403 * @core: device to send the command to
404 * @info:  struct si476x_func_info to fill all the information
405 *         returned by the command
406 *
407 * The command requests the firmware and patch version for currently
408 * loaded firmware (dependent on the function of the device FM/AM/WB)
409 *
410 * Function returns 0 on succsess and negative error code on
411 * failure
412 */
413int si476x_core_cmd_func_info(struct si476x_core *core,
414			      struct si476x_func_info *info)
415{
416	int err;
417	u8  resp[CMD_FUNC_INFO_NRESP];
418
419	err = si476x_core_send_command(core, CMD_FUNC_INFO,
420				       NULL, 0,
421				       resp, ARRAY_SIZE(resp),
422				       SI476X_DEFAULT_TIMEOUT);
423
424	info->firmware.major    = resp[1];
425	info->firmware.minor[0] = resp[2];
426	info->firmware.minor[1] = resp[3];
427
428	info->patch_id = ((u16) resp[4] << 8) | resp[5];
429	info->func     = resp[6];
430
431	return err;
432}
433EXPORT_SYMBOL_GPL(si476x_core_cmd_func_info);
434
435/**
436 * si476x_cmd_set_property() - send 'SET_PROPERTY' command to the device
437 * @core:    device to send the command to
438 * @property: property address
439 * @value:    property value
440 *
441 * Function returns 0 on succsess and negative error code on
442 * failure
443 */
444int si476x_core_cmd_set_property(struct si476x_core *core,
445				 u16 property, u16 value)
446{
447	u8       resp[CMD_SET_PROPERTY_NRESP];
448	const u8 args[CMD_SET_PROPERTY_NARGS] = {
449		0x00,
450		msb(property),
451		lsb(property),
452		msb(value),
453		lsb(value),
454	};
455
456	return si476x_core_send_command(core, CMD_SET_PROPERTY,
457					args, ARRAY_SIZE(args),
458					resp, ARRAY_SIZE(resp),
459					SI476X_DEFAULT_TIMEOUT);
460}
461EXPORT_SYMBOL_GPL(si476x_core_cmd_set_property);
462
463/**
464 * si476x_cmd_get_property() - send 'GET_PROPERTY' command to the device
465 * @core:    device to send the command to
466 * @property: property address
467 *
468 * Function return the value of property as u16 on success or a
469 * negative error on failure
470 */
471int si476x_core_cmd_get_property(struct si476x_core *core, u16 property)
472{
473	int err;
474	u8       resp[CMD_GET_PROPERTY_NRESP];
475	const u8 args[CMD_GET_PROPERTY_NARGS] = {
476		0x00,
477		msb(property),
478		lsb(property),
479	};
480
481	err = si476x_core_send_command(core, CMD_GET_PROPERTY,
482				       args, ARRAY_SIZE(args),
483				       resp, ARRAY_SIZE(resp),
484				       SI476X_DEFAULT_TIMEOUT);
485	if (err < 0)
486		return err;
487	else
488		return get_unaligned_be16(resp + 2);
489}
490EXPORT_SYMBOL_GPL(si476x_core_cmd_get_property);
491
492/**
493 * si476x_cmd_dig_audio_pin_cfg() - send 'DIG_AUDIO_PIN_CFG' command to
494 * the device
495 * @core: device to send the command to
496 * @dclk:  DCLK pin function configuration:
497 *	   #SI476X_DCLK_NOOP     - do not modify the behaviour
498 *         #SI476X_DCLK_TRISTATE - put the pin in tristate condition,
499 *                                 enable 1MOhm pulldown
500 *         #SI476X_DCLK_DAUDIO   - set the pin to be a part of digital
501 *                                 audio interface
502 * @dfs:   DFS pin function configuration:
503 *         #SI476X_DFS_NOOP      - do not modify the behaviour
504 *         #SI476X_DFS_TRISTATE  - put the pin in tristate condition,
505 *                             enable 1MOhm pulldown
506 *      SI476X_DFS_DAUDIO    - set the pin to be a part of digital
507 *                             audio interface
508 * @dout - DOUT pin function configuration:
509 *      SI476X_DOUT_NOOP       - do not modify the behaviour
510 *      SI476X_DOUT_TRISTATE   - put the pin in tristate condition,
511 *                               enable 1MOhm pulldown
512 *      SI476X_DOUT_I2S_OUTPUT - set this pin to be digital out on I2S
513 *                               port 1
514 *      SI476X_DOUT_I2S_INPUT  - set this pin to be digital in on I2S
515 *                               port 1
516 * @xout - XOUT pin function configuration:
517 *	SI476X_XOUT_NOOP        - do not modify the behaviour
518 *      SI476X_XOUT_TRISTATE    - put the pin in tristate condition,
519 *                                enable 1MOhm pulldown
520 *      SI476X_XOUT_I2S_INPUT   - set this pin to be digital in on I2S
521 *                                port 1
522 *      SI476X_XOUT_MODE_SELECT - set this pin to be the input that
523 *                                selects the mode of the I2S audio
524 *                                combiner (analog or HD)
525 *                                [SI4761/63/65/67 Only]
526 *
527 * Function returns 0 on success and negative error code on failure
528 */
529int si476x_core_cmd_dig_audio_pin_cfg(struct  si476x_core *core,
530				      enum si476x_dclk_config dclk,
531				      enum si476x_dfs_config  dfs,
532				      enum si476x_dout_config dout,
533				      enum si476x_xout_config xout)
534{
535	u8       resp[CMD_DIG_AUDIO_PIN_CFG_NRESP];
536	const u8 args[CMD_DIG_AUDIO_PIN_CFG_NARGS] = {
537		PIN_CFG_BYTE(dclk),
538		PIN_CFG_BYTE(dfs),
539		PIN_CFG_BYTE(dout),
540		PIN_CFG_BYTE(xout),
541	};
542
543	return si476x_core_send_command(core, CMD_DIG_AUDIO_PIN_CFG,
544					args, ARRAY_SIZE(args),
545					resp, ARRAY_SIZE(resp),
546					SI476X_DEFAULT_TIMEOUT);
547}
548EXPORT_SYMBOL_GPL(si476x_core_cmd_dig_audio_pin_cfg);
549
550/**
551 * si476x_cmd_zif_pin_cfg - send 'ZIF_PIN_CFG_COMMAND'
552 * @core - device to send the command to
553 * @iqclk - IQCL pin function configuration:
554 *       SI476X_IQCLK_NOOP     - do not modify the behaviour
555 *       SI476X_IQCLK_TRISTATE - put the pin in tristate condition,
556 *                               enable 1MOhm pulldown
557 *       SI476X_IQCLK_IQ       - set pin to be a part of I/Q interace
558 *                               in master mode
559 * @iqfs - IQFS pin function configuration:
560 *       SI476X_IQFS_NOOP     - do not modify the behaviour
561 *       SI476X_IQFS_TRISTATE - put the pin in tristate condition,
562 *                              enable 1MOhm pulldown
563 *       SI476X_IQFS_IQ       - set pin to be a part of I/Q interace
564 *                              in master mode
565 * @iout - IOUT pin function configuration:
566 *       SI476X_IOUT_NOOP     - do not modify the behaviour
567 *       SI476X_IOUT_TRISTATE - put the pin in tristate condition,
568 *                              enable 1MOhm pulldown
569 *       SI476X_IOUT_OUTPUT   - set pin to be I out
570 * @qout - QOUT pin function configuration:
571 *       SI476X_QOUT_NOOP     - do not modify the behaviour
572 *       SI476X_QOUT_TRISTATE - put the pin in tristate condition,
573 *                              enable 1MOhm pulldown
574 *       SI476X_QOUT_OUTPUT   - set pin to be Q out
575 *
576 * Function returns 0 on success and negative error code on failure
577 */
578int si476x_core_cmd_zif_pin_cfg(struct si476x_core *core,
579				enum si476x_iqclk_config iqclk,
580				enum si476x_iqfs_config iqfs,
581				enum si476x_iout_config iout,
582				enum si476x_qout_config qout)
583{
584	u8       resp[CMD_ZIF_PIN_CFG_NRESP];
585	const u8 args[CMD_ZIF_PIN_CFG_NARGS] = {
586		PIN_CFG_BYTE(iqclk),
587		PIN_CFG_BYTE(iqfs),
588		PIN_CFG_BYTE(iout),
589		PIN_CFG_BYTE(qout),
590	};
591
592	return si476x_core_send_command(core, CMD_ZIF_PIN_CFG,
593					args, ARRAY_SIZE(args),
594					resp, ARRAY_SIZE(resp),
595					SI476X_DEFAULT_TIMEOUT);
596}
597EXPORT_SYMBOL_GPL(si476x_core_cmd_zif_pin_cfg);
598
599/**
600 * si476x_cmd_ic_link_gpo_ctl_pin_cfg - send
601 * 'IC_LINK_GPIO_CTL_PIN_CFG' comand to the device
602 * @core - device to send the command to
603 * @icin - ICIN pin function configuration:
604 *      SI476X_ICIN_NOOP      - do not modify the behaviour
605 *      SI476X_ICIN_TRISTATE  - put the pin in tristate condition,
606 *                              enable 1MOhm pulldown
607 *      SI476X_ICIN_GPO1_HIGH - set pin to be an output, drive it high
608 *      SI476X_ICIN_GPO1_LOW  - set pin to be an output, drive it low
609 *      SI476X_ICIN_IC_LINK   - set the pin to be a part of Inter-Chip link
610 * @icip - ICIP pin function configuration:
611 *      SI476X_ICIP_NOOP      - do not modify the behaviour
612 *      SI476X_ICIP_TRISTATE  - put the pin in tristate condition,
613 *                              enable 1MOhm pulldown
614 *      SI476X_ICIP_GPO1_HIGH - set pin to be an output, drive it high
615 *      SI476X_ICIP_GPO1_LOW  - set pin to be an output, drive it low
616 *      SI476X_ICIP_IC_LINK   - set the pin to be a part of Inter-Chip link
617 * @icon - ICON pin function configuration:
618 *      SI476X_ICON_NOOP     - do not modify the behaviour
619 *      SI476X_ICON_TRISTATE - put the pin in tristate condition,
620 *                             enable 1MOhm pulldown
621 *      SI476X_ICON_I2S      - set the pin to be a part of audio
622 *                             interface in slave mode (DCLK)
623 *      SI476X_ICON_IC_LINK  - set the pin to be a part of Inter-Chip link
624 * @icop - ICOP pin function configuration:
625 *      SI476X_ICOP_NOOP     - do not modify the behaviour
626 *      SI476X_ICOP_TRISTATE - put the pin in tristate condition,
627 *                             enable 1MOhm pulldown
628 *      SI476X_ICOP_I2S      - set the pin to be a part of audio
629 *                             interface in slave mode (DOUT)
630 *                             [Si4761/63/65/67 Only]
631 *      SI476X_ICOP_IC_LINK  - set the pin to be a part of Inter-Chip link
632 *
633 * Function returns 0 on success and negative error code on failure
634 */
635int si476x_core_cmd_ic_link_gpo_ctl_pin_cfg(struct si476x_core *core,
636					    enum si476x_icin_config icin,
637					    enum si476x_icip_config icip,
638					    enum si476x_icon_config icon,
639					    enum si476x_icop_config icop)
640{
641	u8       resp[CMD_IC_LINK_GPO_CTL_PIN_CFG_NRESP];
642	const u8 args[CMD_IC_LINK_GPO_CTL_PIN_CFG_NARGS] = {
643		PIN_CFG_BYTE(icin),
644		PIN_CFG_BYTE(icip),
645		PIN_CFG_BYTE(icon),
646		PIN_CFG_BYTE(icop),
647	};
648
649	return si476x_core_send_command(core, CMD_IC_LINK_GPO_CTL_PIN_CFG,
650					args, ARRAY_SIZE(args),
651					resp, ARRAY_SIZE(resp),
652					SI476X_DEFAULT_TIMEOUT);
653}
654EXPORT_SYMBOL_GPL(si476x_core_cmd_ic_link_gpo_ctl_pin_cfg);
655
656/**
657 * si476x_cmd_ana_audio_pin_cfg - send 'ANA_AUDIO_PIN_CFG' to the
658 * device
659 * @core - device to send the command to
660 * @lrout - LROUT pin function configuration:
661 *       SI476X_LROUT_NOOP     - do not modify the behaviour
662 *       SI476X_LROUT_TRISTATE - put the pin in tristate condition,
663 *                               enable 1MOhm pulldown
664 *       SI476X_LROUT_AUDIO    - set pin to be audio output
665 *       SI476X_LROUT_MPX      - set pin to be MPX output
666 *
667 * Function returns 0 on success and negative error code on failure
668 */
669int si476x_core_cmd_ana_audio_pin_cfg(struct si476x_core *core,
670				      enum si476x_lrout_config lrout)
671{
672	u8       resp[CMD_ANA_AUDIO_PIN_CFG_NRESP];
673	const u8 args[CMD_ANA_AUDIO_PIN_CFG_NARGS] = {
674		PIN_CFG_BYTE(lrout),
675	};
676
677	return si476x_core_send_command(core, CMD_ANA_AUDIO_PIN_CFG,
678					args, ARRAY_SIZE(args),
679					resp, ARRAY_SIZE(resp),
680					SI476X_DEFAULT_TIMEOUT);
681}
682EXPORT_SYMBOL_GPL(si476x_core_cmd_ana_audio_pin_cfg);
683
684
685/**
686 * si476x_cmd_intb_pin_cfg - send 'INTB_PIN_CFG' command to the device
687 * @core - device to send the command to
688 * @intb - INTB pin function configuration:
689 *      SI476X_INTB_NOOP     - do not modify the behaviour
690 *      SI476X_INTB_TRISTATE - put the pin in tristate condition,
691 *                             enable 1MOhm pulldown
692 *      SI476X_INTB_DAUDIO   - set pin to be a part of digital
693 *                             audio interface in slave mode
694 *      SI476X_INTB_IRQ      - set pin to be an interrupt request line
695 * @a1 - A1 pin function configuration:
696 *      SI476X_A1_NOOP     - do not modify the behaviour
697 *      SI476X_A1_TRISTATE - put the pin in tristate condition,
698 *                           enable 1MOhm pulldown
699 *      SI476X_A1_IRQ      - set pin to be an interrupt request line
700 *
701 * Function returns 0 on success and negative error code on failure
702 */
703static int si476x_core_cmd_intb_pin_cfg_a10(struct si476x_core *core,
704					    enum si476x_intb_config intb,
705					    enum si476x_a1_config a1)
706{
707	u8       resp[CMD_INTB_PIN_CFG_A10_NRESP];
708	const u8 args[CMD_INTB_PIN_CFG_NARGS] = {
709		PIN_CFG_BYTE(intb),
710		PIN_CFG_BYTE(a1),
711	};
712
713	return si476x_core_send_command(core, CMD_INTB_PIN_CFG,
714					args, ARRAY_SIZE(args),
715					resp, ARRAY_SIZE(resp),
716					SI476X_DEFAULT_TIMEOUT);
717}
718
719static int si476x_core_cmd_intb_pin_cfg_a20(struct si476x_core *core,
720					    enum si476x_intb_config intb,
721					    enum si476x_a1_config a1)
722{
723	u8       resp[CMD_INTB_PIN_CFG_A20_NRESP];
724	const u8 args[CMD_INTB_PIN_CFG_NARGS] = {
725		PIN_CFG_BYTE(intb),
726		PIN_CFG_BYTE(a1),
727	};
728
729	return si476x_core_send_command(core, CMD_INTB_PIN_CFG,
730					args, ARRAY_SIZE(args),
731					resp, ARRAY_SIZE(resp),
732					SI476X_DEFAULT_TIMEOUT);
733}
734
735
736
737/**
738 * si476x_cmd_am_rsq_status - send 'AM_RSQ_STATUS' command to the
739 * device
740 * @core  - device to send the command to
741 * @rsqack - if set command clears RSQINT, SNRINT, SNRLINT, RSSIHINT,
742 *           RSSSILINT, BLENDINT, MULTHINT and MULTLINT
743 * @attune - when set the values in the status report are the values
744 *           that were calculated at tune
745 * @cancel - abort ongoing seek/tune opertation
746 * @stcack - clear the STCINT bin in status register
747 * @report - all signal quality information retured by the command
748 *           (if NULL then the output of the command is ignored)
749 *
750 * Function returns 0 on success and negative error code on failure
751 */
752int si476x_core_cmd_am_rsq_status(struct si476x_core *core,
753				  struct si476x_rsq_status_args *rsqargs,
754				  struct si476x_rsq_status_report *report)
755{
756	int err;
757	u8       resp[CMD_AM_RSQ_STATUS_NRESP];
758	const u8 args[CMD_AM_RSQ_STATUS_NARGS] = {
759		rsqargs->rsqack << 3 | rsqargs->attune << 2 |
760		rsqargs->cancel << 1 | rsqargs->stcack,
761	};
762
763	err = si476x_core_send_command(core, CMD_AM_RSQ_STATUS,
764				       args, ARRAY_SIZE(args),
765				       resp, ARRAY_SIZE(resp),
766				       SI476X_DEFAULT_TIMEOUT);
767	/*
768	 * Besides getting received signal quality information this
769	 * command can be used to just acknowledge different interrupt
770	 * flags in those cases it is useless to copy and parse
771	 * received data so user can pass NULL, and thus avoid
772	 * unnecessary copying.
773	 */
774	if (!report)
775		return err;
776
777	report->snrhint		= 0x08 & resp[1];
778	report->snrlint		= 0x04 & resp[1];
779	report->rssihint	= 0x02 & resp[1];
780	report->rssilint	= 0x01 & resp[1];
781
782	report->bltf		= 0x80 & resp[2];
783	report->snr_ready	= 0x20 & resp[2];
784	report->rssiready	= 0x08 & resp[2];
785	report->afcrl		= 0x02 & resp[2];
786	report->valid		= 0x01 & resp[2];
787
788	report->readfreq	= get_unaligned_be16(resp + 3);
789	report->freqoff		= resp[5];
790	report->rssi		= resp[6];
791	report->snr		= resp[7];
792	report->lassi		= resp[9];
793	report->hassi		= resp[10];
794	report->mult		= resp[11];
795	report->dev		= resp[12];
796
797	return err;
798}
799EXPORT_SYMBOL_GPL(si476x_core_cmd_am_rsq_status);
800
801int si476x_core_cmd_fm_acf_status(struct si476x_core *core,
802			     struct si476x_acf_status_report *report)
803{
804	int err;
805	u8       resp[CMD_FM_ACF_STATUS_NRESP];
806	const u8 args[CMD_FM_ACF_STATUS_NARGS] = {
807		0x0,
808	};
809
810	if (!report)
811		return -EINVAL;
812
813	err = si476x_core_send_command(core, CMD_FM_ACF_STATUS,
814				       args, ARRAY_SIZE(args),
815				       resp, ARRAY_SIZE(resp),
816				       SI476X_DEFAULT_TIMEOUT);
817	if (err < 0)
818		return err;
819
820	report->blend_int	= resp[1] & SI476X_ACF_BLEND_INT;
821	report->hblend_int	= resp[1] & SI476X_ACF_HIBLEND_INT;
822	report->hicut_int	= resp[1] & SI476X_ACF_HICUT_INT;
823	report->chbw_int	= resp[1] & SI476X_ACF_CHBW_INT;
824	report->softmute_int	= resp[1] & SI476X_ACF_SOFTMUTE_INT;
825	report->smute		= resp[2] & SI476X_ACF_SMUTE;
826	report->smattn		= resp[3] & SI476X_ACF_SMATTN;
827	report->chbw		= resp[4];
828	report->hicut		= resp[5];
829	report->hiblend		= resp[6];
830	report->pilot		= resp[7] & SI476X_ACF_PILOT;
831	report->stblend		= resp[7] & SI476X_ACF_STBLEND;
832
833	return err;
834}
835EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_acf_status);
836
837int si476x_core_cmd_am_acf_status(struct si476x_core *core,
838				  struct si476x_acf_status_report *report)
839{
840	int err;
841	u8       resp[CMD_AM_ACF_STATUS_NRESP];
842	const u8 args[CMD_AM_ACF_STATUS_NARGS] = {
843		0x0,
844	};
845
846	if (!report)
847		return -EINVAL;
848
849	err = si476x_core_send_command(core, CMD_AM_ACF_STATUS,
850				       args, ARRAY_SIZE(args),
851				       resp, ARRAY_SIZE(resp),
852				       SI476X_DEFAULT_TIMEOUT);
853	if (err < 0)
854		return err;
855
856	report->blend_int	= resp[1] & SI476X_ACF_BLEND_INT;
857	report->hblend_int	= resp[1] & SI476X_ACF_HIBLEND_INT;
858	report->hicut_int	= resp[1] & SI476X_ACF_HICUT_INT;
859	report->chbw_int	= resp[1] & SI476X_ACF_CHBW_INT;
860	report->softmute_int	= resp[1] & SI476X_ACF_SOFTMUTE_INT;
861	report->smute		= resp[2] & SI476X_ACF_SMUTE;
862	report->smattn		= resp[3] & SI476X_ACF_SMATTN;
863	report->chbw		= resp[4];
864	report->hicut		= resp[5];
865
866	return err;
867}
868EXPORT_SYMBOL_GPL(si476x_core_cmd_am_acf_status);
869
870
871/**
872 * si476x_cmd_fm_seek_start - send 'FM_SEEK_START' command to the
873 * device
874 * @core  - device to send the command to
875 * @seekup - if set the direction of the search is 'up'
876 * @wrap   - if set seek wraps when hitting band limit
877 *
878 * This function begins search for a valid station. The station is
879 * considered valid when 'FM_VALID_SNR_THRESHOLD' and
880 * 'FM_VALID_RSSI_THRESHOLD' and 'FM_VALID_MAX_TUNE_ERROR' criteria
881 * are met.
882} *
883 * Function returns 0 on success and negative error code on failure
884 */
885int si476x_core_cmd_fm_seek_start(struct si476x_core *core,
886				  bool seekup, bool wrap)
887{
888	u8       resp[CMD_FM_SEEK_START_NRESP];
889	const u8 args[CMD_FM_SEEK_START_NARGS] = {
890		seekup << 3 | wrap << 2,
891	};
892
893	return si476x_cmd_tune_seek_freq(core, CMD_FM_SEEK_START,
894					 args, sizeof(args),
895					 resp, sizeof(resp));
896}
897EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_seek_start);
898
899/**
900 * si476x_cmd_fm_rds_status - send 'FM_RDS_STATUS' command to the
901 * device
902 * @core - device to send the command to
903 * @status_only - if set the data is not removed from RDSFIFO,
904 *                RDSFIFOUSED is not decremented and data in all the
905 *                rest RDS data contains the last valid info received
906 * @mtfifo if set the command clears RDS receive FIFO
907 * @intack if set the command clards the RDSINT bit.
908 *
909 * Function returns 0 on success and negative error code on failure
910 */
911int si476x_core_cmd_fm_rds_status(struct si476x_core *core,
912				  bool status_only,
913				  bool mtfifo,
914				  bool intack,
915				  struct si476x_rds_status_report *report)
916{
917	int err;
918	u8       resp[CMD_FM_RDS_STATUS_NRESP];
919	const u8 args[CMD_FM_RDS_STATUS_NARGS] = {
920		status_only << 2 | mtfifo << 1 | intack,
921	};
922
923	err = si476x_core_send_command(core, CMD_FM_RDS_STATUS,
924				       args, ARRAY_SIZE(args),
925				       resp, ARRAY_SIZE(resp),
926				       SI476X_DEFAULT_TIMEOUT);
927	/*
928	 * Besides getting RDS status information this command can be
929	 * used to just acknowledge different interrupt flags in those
930	 * cases it is useless to copy and parse received data so user
931	 * can pass NULL, and thus avoid unnecessary copying.
932	 */
933	if (err < 0 || report == NULL)
934		return err;
935
936	report->rdstpptyint	= 0x10 & resp[1];
937	report->rdspiint	= 0x08 & resp[1];
938	report->rdssyncint	= 0x02 & resp[1];
939	report->rdsfifoint	= 0x01 & resp[1];
940
941	report->tpptyvalid	= 0x10 & resp[2];
942	report->pivalid		= 0x08 & resp[2];
943	report->rdssync		= 0x02 & resp[2];
944	report->rdsfifolost	= 0x01 & resp[2];
945
946	report->tp		= 0x20 & resp[3];
947	report->pty		= 0x1f & resp[3];
948
949	report->pi		= get_unaligned_be16(resp + 4);
950	report->rdsfifoused	= resp[6];
951
952	report->ble[V4L2_RDS_BLOCK_A]	= 0xc0 & resp[7];
953	report->ble[V4L2_RDS_BLOCK_B]	= 0x30 & resp[7];
954	report->ble[V4L2_RDS_BLOCK_C]	= 0x0c & resp[7];
955	report->ble[V4L2_RDS_BLOCK_D]	= 0x03 & resp[7];
956
957	report->rds[V4L2_RDS_BLOCK_A].block = V4L2_RDS_BLOCK_A;
958	report->rds[V4L2_RDS_BLOCK_A].msb = resp[8];
959	report->rds[V4L2_RDS_BLOCK_A].lsb = resp[9];
960
961	report->rds[V4L2_RDS_BLOCK_B].block = V4L2_RDS_BLOCK_B;
962	report->rds[V4L2_RDS_BLOCK_B].msb = resp[10];
963	report->rds[V4L2_RDS_BLOCK_B].lsb = resp[11];
964
965	report->rds[V4L2_RDS_BLOCK_C].block = V4L2_RDS_BLOCK_C;
966	report->rds[V4L2_RDS_BLOCK_C].msb = resp[12];
967	report->rds[V4L2_RDS_BLOCK_C].lsb = resp[13];
968
969	report->rds[V4L2_RDS_BLOCK_D].block = V4L2_RDS_BLOCK_D;
970	report->rds[V4L2_RDS_BLOCK_D].msb = resp[14];
971	report->rds[V4L2_RDS_BLOCK_D].lsb = resp[15];
972
973	return err;
974}
975EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_rds_status);
976
977int si476x_core_cmd_fm_rds_blockcount(struct si476x_core *core,
978				bool clear,
979				struct si476x_rds_blockcount_report *report)
980{
981	int err;
982	u8       resp[CMD_FM_RDS_BLOCKCOUNT_NRESP];
983	const u8 args[CMD_FM_RDS_BLOCKCOUNT_NARGS] = {
984		clear,
985	};
986
987	if (!report)
988		return -EINVAL;
989
990	err = si476x_core_send_command(core, CMD_FM_RDS_BLOCKCOUNT,
991				       args, ARRAY_SIZE(args),
992				       resp, ARRAY_SIZE(resp),
993				       SI476X_DEFAULT_TIMEOUT);
994
995	if (!err) {
996		report->expected	= get_unaligned_be16(resp + 2);
997		report->received	= get_unaligned_be16(resp + 4);
998		report->uncorrectable	= get_unaligned_be16(resp + 6);
999	}
1000
1001	return err;
1002}
1003EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_rds_blockcount);
1004
1005int si476x_core_cmd_fm_phase_diversity(struct si476x_core *core,
1006				       enum si476x_phase_diversity_mode mode)
1007{
1008	u8       resp[CMD_FM_PHASE_DIVERSITY_NRESP];
1009	const u8 args[CMD_FM_PHASE_DIVERSITY_NARGS] = {
1010		mode & 0x07,
1011	};
1012
1013	return si476x_core_send_command(core, CMD_FM_PHASE_DIVERSITY,
1014					args, ARRAY_SIZE(args),
1015					resp, ARRAY_SIZE(resp),
1016					SI476X_DEFAULT_TIMEOUT);
1017}
1018EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_phase_diversity);
1019/**
1020 * si476x_core_cmd_fm_phase_div_status() - get the phase diversity
1021 * status
1022 *
1023 * @core: si476x device
1024 *
1025 * NOTE caller must hold core lock
1026 *
1027 * Function returns the value of the status bit in case of success and
1028 * negative error code in case of failre.
1029 */
1030int si476x_core_cmd_fm_phase_div_status(struct si476x_core *core)
1031{
1032	int err;
1033	u8 resp[CMD_FM_PHASE_DIV_STATUS_NRESP];
1034
1035	err = si476x_core_send_command(core, CMD_FM_PHASE_DIV_STATUS,
1036				       NULL, 0,
1037				       resp, ARRAY_SIZE(resp),
1038				       SI476X_DEFAULT_TIMEOUT);
1039
1040	return (err < 0) ? err : resp[1];
1041}
1042EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_phase_div_status);
1043
1044
1045/**
1046 * si476x_cmd_am_seek_start - send 'FM_SEEK_START' command to the
1047 * device
1048 * @core  - device to send the command to
1049 * @seekup - if set the direction of the search is 'up'
1050 * @wrap   - if set seek wraps when hitting band limit
1051 *
1052 * This function begins search for a valid station. The station is
1053 * considered valid when 'FM_VALID_SNR_THRESHOLD' and
1054 * 'FM_VALID_RSSI_THRESHOLD' and 'FM_VALID_MAX_TUNE_ERROR' criteria
1055 * are met.
1056 *
1057 * Function returns 0 on success and negative error code on failure
1058 */
1059int si476x_core_cmd_am_seek_start(struct si476x_core *core,
1060				  bool seekup, bool wrap)
1061{
1062	u8       resp[CMD_AM_SEEK_START_NRESP];
1063	const u8 args[CMD_AM_SEEK_START_NARGS] = {
1064		seekup << 3 | wrap << 2,
1065	};
1066
1067	return si476x_cmd_tune_seek_freq(core,  CMD_AM_SEEK_START,
1068					 args, sizeof(args),
1069					 resp, sizeof(resp));
1070}
1071EXPORT_SYMBOL_GPL(si476x_core_cmd_am_seek_start);
1072
1073
1074
1075static int si476x_core_cmd_power_up_a10(struct si476x_core *core,
1076					struct si476x_power_up_args *puargs)
1077{
1078	u8       resp[CMD_POWER_UP_A10_NRESP];
1079	const bool intsel = (core->pinmux.a1 == SI476X_A1_IRQ);
1080	const bool ctsen  = (core->client->irq != 0);
1081	const u8 args[CMD_POWER_UP_A10_NARGS] = {
1082		0xF7,		/* Reserved, always 0xF7 */
1083		0x3F & puargs->xcload,	/* First two bits are reserved to be
1084				 * zeros */
1085		ctsen << 7 | intsel << 6 | 0x07, /* Last five bits
1086						   * are reserved to
1087						   * be written as 0x7 */
1088		puargs->func << 4 | puargs->freq,
1089		0x11,		/* Reserved, always 0x11 */
1090	};
1091
1092	return si476x_core_send_command(core, CMD_POWER_UP,
1093					args, ARRAY_SIZE(args),
1094					resp, ARRAY_SIZE(resp),
1095					SI476X_TIMEOUT_POWER_UP);
1096}
1097
1098static int si476x_core_cmd_power_up_a20(struct si476x_core *core,
1099				 struct si476x_power_up_args *puargs)
1100{
1101	u8       resp[CMD_POWER_UP_A20_NRESP];
1102	const bool intsel = (core->pinmux.a1 == SI476X_A1_IRQ);
1103	const bool ctsen  = (core->client->irq != 0);
1104	const u8 args[CMD_POWER_UP_A20_NARGS] = {
1105		puargs->ibias6x << 7 | puargs->xstart,
1106		0x3F & puargs->xcload,	/* First two bits are reserved to be
1107					 * zeros */
1108		ctsen << 7 | intsel << 6 | puargs->fastboot << 5 |
1109		puargs->xbiashc << 3 | puargs->xbias,
1110		puargs->func << 4 | puargs->freq,
1111		0x10 | puargs->xmode,
1112	};
1113
1114	return si476x_core_send_command(core, CMD_POWER_UP,
1115					args, ARRAY_SIZE(args),
1116					resp, ARRAY_SIZE(resp),
1117					SI476X_TIMEOUT_POWER_UP);
1118}
1119
1120static int si476x_core_cmd_power_down_a10(struct si476x_core *core,
1121					  struct si476x_power_down_args *pdargs)
1122{
1123	u8 resp[CMD_POWER_DOWN_A10_NRESP];
1124
1125	return si476x_core_send_command(core, CMD_POWER_DOWN,
1126					NULL, 0,
1127					resp, ARRAY_SIZE(resp),
1128					SI476X_DEFAULT_TIMEOUT);
1129}
1130
1131static int si476x_core_cmd_power_down_a20(struct si476x_core *core,
1132					  struct si476x_power_down_args *pdargs)
1133{
1134	u8 resp[CMD_POWER_DOWN_A20_NRESP];
1135	const u8 args[CMD_POWER_DOWN_A20_NARGS] = {
1136		pdargs->xosc,
1137	};
1138	return si476x_core_send_command(core, CMD_POWER_DOWN,
1139					args, ARRAY_SIZE(args),
1140					resp, ARRAY_SIZE(resp),
1141					SI476X_DEFAULT_TIMEOUT);
1142}
1143
1144static int si476x_core_cmd_am_tune_freq_a10(struct si476x_core *core,
1145					struct si476x_tune_freq_args *tuneargs)
1146{
1147
1148	const int am_freq = tuneargs->freq;
1149	u8       resp[CMD_AM_TUNE_FREQ_NRESP];
1150	const u8 args[CMD_AM_TUNE_FREQ_NARGS] = {
1151		(tuneargs->hd << 6),
1152		msb(am_freq),
1153		lsb(am_freq),
1154	};
1155
1156	return si476x_cmd_tune_seek_freq(core, CMD_AM_TUNE_FREQ, args,
1157					 sizeof(args),
1158					 resp, sizeof(resp));
1159}
1160
1161static int si476x_core_cmd_am_tune_freq_a20(struct si476x_core *core,
1162					struct si476x_tune_freq_args *tuneargs)
1163{
1164	const int am_freq = tuneargs->freq;
1165	u8       resp[CMD_AM_TUNE_FREQ_NRESP];
1166	const u8 args[CMD_AM_TUNE_FREQ_NARGS] = {
1167		(tuneargs->zifsr << 6) | (tuneargs->injside & 0x03),
1168		msb(am_freq),
1169		lsb(am_freq),
1170	};
1171
1172	return si476x_cmd_tune_seek_freq(core, CMD_AM_TUNE_FREQ,
1173					 args, sizeof(args),
1174					 resp, sizeof(resp));
1175}
1176
1177static int si476x_core_cmd_fm_rsq_status_a10(struct si476x_core *core,
1178					struct si476x_rsq_status_args *rsqargs,
1179					struct si476x_rsq_status_report *report)
1180{
1181	int err;
1182	u8       resp[CMD_FM_RSQ_STATUS_A10_NRESP];
1183	const u8 args[CMD_FM_RSQ_STATUS_A10_NARGS] = {
1184		rsqargs->rsqack << 3 | rsqargs->attune << 2 |
1185		rsqargs->cancel << 1 | rsqargs->stcack,
1186	};
1187
1188	err = si476x_core_send_command(core, CMD_FM_RSQ_STATUS,
1189				       args, ARRAY_SIZE(args),
1190				       resp, ARRAY_SIZE(resp),
1191				       SI476X_DEFAULT_TIMEOUT);
1192	/*
1193	 * Besides getting received signal quality information this
1194	 * command can be used to just acknowledge different interrupt
1195	 * flags in those cases it is useless to copy and parse
1196	 * received data so user can pass NULL, and thus avoid
1197	 * unnecessary copying.
1198	 */
1199	if (err < 0 || report == NULL)
1200		return err;
1201
1202	report->multhint	= 0x80 & resp[1];
1203	report->multlint	= 0x40 & resp[1];
1204	report->snrhint		= 0x08 & resp[1];
1205	report->snrlint		= 0x04 & resp[1];
1206	report->rssihint	= 0x02 & resp[1];
1207	report->rssilint	= 0x01 & resp[1];
1208
1209	report->bltf		= 0x80 & resp[2];
1210	report->snr_ready	= 0x20 & resp[2];
1211	report->rssiready	= 0x08 & resp[2];
1212	report->afcrl		= 0x02 & resp[2];
1213	report->valid		= 0x01 & resp[2];
1214
1215	report->readfreq	= get_unaligned_be16(resp + 3);
1216	report->freqoff		= resp[5];
1217	report->rssi		= resp[6];
1218	report->snr		= resp[7];
1219	report->lassi		= resp[9];
1220	report->hassi		= resp[10];
1221	report->mult		= resp[11];
1222	report->dev		= resp[12];
1223	report->readantcap	= get_unaligned_be16(resp + 13);
1224	report->assi		= resp[15];
1225	report->usn		= resp[16];
1226
1227	return err;
1228}
1229
1230static int si476x_core_cmd_fm_rsq_status_a20(struct si476x_core *core,
1231				     struct si476x_rsq_status_args *rsqargs,
1232				     struct si476x_rsq_status_report *report)
1233{
1234	int err;
1235	u8       resp[CMD_FM_RSQ_STATUS_A10_NRESP];
1236	const u8 args[CMD_FM_RSQ_STATUS_A30_NARGS] = {
1237		rsqargs->primary << 4 | rsqargs->rsqack << 3 |
1238		rsqargs->attune  << 2 | rsqargs->cancel << 1 |
1239		rsqargs->stcack,
1240	};
1241
1242	err = si476x_core_send_command(core, CMD_FM_RSQ_STATUS,
1243				       args, ARRAY_SIZE(args),
1244				       resp, ARRAY_SIZE(resp),
1245				       SI476X_DEFAULT_TIMEOUT);
1246	/*
1247	 * Besides getting received signal quality information this
1248	 * command can be used to just acknowledge different interrupt
1249	 * flags in those cases it is useless to copy and parse
1250	 * received data so user can pass NULL, and thus avoid
1251	 * unnecessary copying.
1252	 */
1253	if (err < 0 || report == NULL)
1254		return err;
1255
1256	report->multhint	= 0x80 & resp[1];
1257	report->multlint	= 0x40 & resp[1];
1258	report->snrhint		= 0x08 & resp[1];
1259	report->snrlint		= 0x04 & resp[1];
1260	report->rssihint	= 0x02 & resp[1];
1261	report->rssilint	= 0x01 & resp[1];
1262
1263	report->bltf		= 0x80 & resp[2];
1264	report->snr_ready	= 0x20 & resp[2];
1265	report->rssiready	= 0x08 & resp[2];
1266	report->afcrl		= 0x02 & resp[2];
1267	report->valid		= 0x01 & resp[2];
1268
1269	report->readfreq	= get_unaligned_be16(resp + 3);
1270	report->freqoff		= resp[5];
1271	report->rssi		= resp[6];
1272	report->snr		= resp[7];
1273	report->lassi		= resp[9];
1274	report->hassi		= resp[10];
1275	report->mult		= resp[11];
1276	report->dev		= resp[12];
1277	report->readantcap	= get_unaligned_be16(resp + 13);
1278	report->assi		= resp[15];
1279	report->usn		= resp[16];
1280
1281	return err;
1282}
1283
1284
1285static int si476x_core_cmd_fm_rsq_status_a30(struct si476x_core *core,
1286					struct si476x_rsq_status_args *rsqargs,
1287					struct si476x_rsq_status_report *report)
1288{
1289	int err;
1290	u8       resp[CMD_FM_RSQ_STATUS_A30_NRESP];
1291	const u8 args[CMD_FM_RSQ_STATUS_A30_NARGS] = {
1292		rsqargs->primary << 4 | rsqargs->rsqack << 3 |
1293		rsqargs->attune << 2 | rsqargs->cancel << 1 |
1294		rsqargs->stcack,
1295	};
1296
1297	err = si476x_core_send_command(core, CMD_FM_RSQ_STATUS,
1298				       args, ARRAY_SIZE(args),
1299				       resp, ARRAY_SIZE(resp),
1300				       SI476X_DEFAULT_TIMEOUT);
1301	/*
1302	 * Besides getting received signal quality information this
1303	 * command can be used to just acknowledge different interrupt
1304	 * flags in those cases it is useless to copy and parse
1305	 * received data so user can pass NULL, and thus avoid
1306	 * unnecessary copying.
1307	 */
1308	if (err < 0 || report == NULL)
1309		return err;
1310
1311	report->multhint	= 0x80 & resp[1];
1312	report->multlint	= 0x40 & resp[1];
1313	report->snrhint		= 0x08 & resp[1];
1314	report->snrlint		= 0x04 & resp[1];
1315	report->rssihint	= 0x02 & resp[1];
1316	report->rssilint	= 0x01 & resp[1];
1317
1318	report->bltf		= 0x80 & resp[2];
1319	report->snr_ready	= 0x20 & resp[2];
1320	report->rssiready	= 0x08 & resp[2];
1321	report->injside         = 0x04 & resp[2];
1322	report->afcrl		= 0x02 & resp[2];
1323	report->valid		= 0x01 & resp[2];
1324
1325	report->readfreq	= get_unaligned_be16(resp + 3);
1326	report->freqoff		= resp[5];
1327	report->rssi		= resp[6];
1328	report->snr		= resp[7];
1329	report->issi		= resp[8];
1330	report->lassi		= resp[9];
1331	report->hassi		= resp[10];
1332	report->mult		= resp[11];
1333	report->dev		= resp[12];
1334	report->readantcap	= get_unaligned_be16(resp + 13);
1335	report->assi		= resp[15];
1336	report->usn		= resp[16];
1337
1338	report->pilotdev	= resp[17];
1339	report->rdsdev		= resp[18];
1340	report->assidev		= resp[19];
1341	report->strongdev	= resp[20];
1342	report->rdspi		= get_unaligned_be16(resp + 21);
1343
1344	return err;
1345}
1346
1347static int si476x_core_cmd_fm_tune_freq_a10(struct si476x_core *core,
1348					struct si476x_tune_freq_args *tuneargs)
1349{
1350	u8       resp[CMD_FM_TUNE_FREQ_NRESP];
1351	const u8 args[CMD_FM_TUNE_FREQ_A10_NARGS] = {
1352		(tuneargs->hd << 6) | (tuneargs->tunemode << 4)
1353		| (tuneargs->smoothmetrics << 2),
1354		msb(tuneargs->freq),
1355		lsb(tuneargs->freq),
1356		msb(tuneargs->antcap),
1357		lsb(tuneargs->antcap)
1358	};
1359
1360	return si476x_cmd_tune_seek_freq(core, CMD_FM_TUNE_FREQ,
1361					 args, sizeof(args),
1362					 resp, sizeof(resp));
1363}
1364
1365static int si476x_core_cmd_fm_tune_freq_a20(struct si476x_core *core,
1366					struct si476x_tune_freq_args *tuneargs)
1367{
1368	u8       resp[CMD_FM_TUNE_FREQ_NRESP];
1369	const u8 args[CMD_FM_TUNE_FREQ_A20_NARGS] = {
1370		(tuneargs->hd << 6) | (tuneargs->tunemode << 4)
1371		|  (tuneargs->smoothmetrics << 2) | (tuneargs->injside),
1372		msb(tuneargs->freq),
1373		lsb(tuneargs->freq),
1374	};
1375
1376	return si476x_cmd_tune_seek_freq(core, CMD_FM_TUNE_FREQ,
1377					 args, sizeof(args),
1378					 resp, sizeof(resp));
1379}
1380
1381static int si476x_core_cmd_agc_status_a20(struct si476x_core *core,
1382					struct si476x_agc_status_report *report)
1383{
1384	int err;
1385	u8 resp[CMD_AGC_STATUS_NRESP_A20];
1386
1387	if (!report)
1388		return -EINVAL;
1389
1390	err = si476x_core_send_command(core, CMD_AGC_STATUS,
1391				       NULL, 0,
1392				       resp, ARRAY_SIZE(resp),
1393				       SI476X_DEFAULT_TIMEOUT);
1394	if (err < 0)
1395		return err;
1396
1397	report->mxhi		= resp[1] & SI476X_AGC_MXHI;
1398	report->mxlo		= resp[1] & SI476X_AGC_MXLO;
1399	report->lnahi		= resp[1] & SI476X_AGC_LNAHI;
1400	report->lnalo		= resp[1] & SI476X_AGC_LNALO;
1401	report->fmagc1		= resp[2];
1402	report->fmagc2		= resp[3];
1403	report->pgagain		= resp[4];
1404	report->fmwblang	= resp[5];
1405
1406	return err;
1407}
1408
1409static int si476x_core_cmd_agc_status_a10(struct si476x_core *core,
1410					struct si476x_agc_status_report *report)
1411{
1412	int err;
1413	u8 resp[CMD_AGC_STATUS_NRESP_A10];
1414
1415	if (!report)
1416		return -EINVAL;
1417
1418	err = si476x_core_send_command(core, CMD_AGC_STATUS,
1419				       NULL, 0,
1420				       resp, ARRAY_SIZE(resp),
1421				       SI476X_DEFAULT_TIMEOUT);
1422	if (err < 0)
1423		return err;
1424
1425	report->mxhi	= resp[1] & SI476X_AGC_MXHI;
1426	report->mxlo	= resp[1] & SI476X_AGC_MXLO;
1427	report->lnahi	= resp[1] & SI476X_AGC_LNAHI;
1428	report->lnalo	= resp[1] & SI476X_AGC_LNALO;
1429
1430	return err;
1431}
1432
1433typedef int (*tune_freq_func_t) (struct si476x_core *core,
1434				 struct si476x_tune_freq_args *tuneargs);
1435
1436static struct {
1437	int (*power_up)(struct si476x_core *,
1438			struct si476x_power_up_args *);
1439	int (*power_down)(struct si476x_core *,
1440			  struct si476x_power_down_args *);
1441
1442	tune_freq_func_t fm_tune_freq;
1443	tune_freq_func_t am_tune_freq;
1444
1445	int (*fm_rsq_status)(struct si476x_core *,
1446			     struct si476x_rsq_status_args *,
1447			     struct si476x_rsq_status_report *);
1448
1449	int (*agc_status)(struct si476x_core *,
1450			  struct si476x_agc_status_report *);
1451	int (*intb_pin_cfg)(struct si476x_core *core,
1452			    enum si476x_intb_config intb,
1453			    enum si476x_a1_config a1);
1454} si476x_cmds_vtable[] = {
1455	[SI476X_REVISION_A10] = {
1456		.power_up	= si476x_core_cmd_power_up_a10,
1457		.power_down	= si476x_core_cmd_power_down_a10,
1458		.fm_tune_freq	= si476x_core_cmd_fm_tune_freq_a10,
1459		.am_tune_freq	= si476x_core_cmd_am_tune_freq_a10,
1460		.fm_rsq_status	= si476x_core_cmd_fm_rsq_status_a10,
1461		.agc_status	= si476x_core_cmd_agc_status_a10,
1462		.intb_pin_cfg   = si476x_core_cmd_intb_pin_cfg_a10,
1463	},
1464	[SI476X_REVISION_A20] = {
1465		.power_up	= si476x_core_cmd_power_up_a20,
1466		.power_down	= si476x_core_cmd_power_down_a20,
1467		.fm_tune_freq	= si476x_core_cmd_fm_tune_freq_a20,
1468		.am_tune_freq	= si476x_core_cmd_am_tune_freq_a20,
1469		.fm_rsq_status	= si476x_core_cmd_fm_rsq_status_a20,
1470		.agc_status	= si476x_core_cmd_agc_status_a20,
1471		.intb_pin_cfg   = si476x_core_cmd_intb_pin_cfg_a20,
1472	},
1473	[SI476X_REVISION_A30] = {
1474		.power_up	= si476x_core_cmd_power_up_a20,
1475		.power_down	= si476x_core_cmd_power_down_a20,
1476		.fm_tune_freq	= si476x_core_cmd_fm_tune_freq_a20,
1477		.am_tune_freq	= si476x_core_cmd_am_tune_freq_a20,
1478		.fm_rsq_status	= si476x_core_cmd_fm_rsq_status_a30,
1479		.agc_status	= si476x_core_cmd_agc_status_a20,
1480		.intb_pin_cfg   = si476x_core_cmd_intb_pin_cfg_a20,
1481	},
1482};
1483
1484int si476x_core_cmd_power_up(struct si476x_core *core,
1485			     struct si476x_power_up_args *args)
1486{
1487	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1488	       core->revision == -1);
1489	return si476x_cmds_vtable[core->revision].power_up(core, args);
1490}
1491EXPORT_SYMBOL_GPL(si476x_core_cmd_power_up);
1492
1493int si476x_core_cmd_power_down(struct si476x_core *core,
1494			       struct si476x_power_down_args *args)
1495{
1496	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1497	       core->revision == -1);
1498	return si476x_cmds_vtable[core->revision].power_down(core, args);
1499}
1500EXPORT_SYMBOL_GPL(si476x_core_cmd_power_down);
1501
1502int si476x_core_cmd_fm_tune_freq(struct si476x_core *core,
1503				 struct si476x_tune_freq_args *args)
1504{
1505	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1506	       core->revision == -1);
1507	return si476x_cmds_vtable[core->revision].fm_tune_freq(core, args);
1508}
1509EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_tune_freq);
1510
1511int si476x_core_cmd_am_tune_freq(struct si476x_core *core,
1512				 struct si476x_tune_freq_args *args)
1513{
1514	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1515	       core->revision == -1);
1516	return si476x_cmds_vtable[core->revision].am_tune_freq(core, args);
1517}
1518EXPORT_SYMBOL_GPL(si476x_core_cmd_am_tune_freq);
1519
1520int si476x_core_cmd_fm_rsq_status(struct si476x_core *core,
1521				  struct si476x_rsq_status_args *args,
1522				  struct si476x_rsq_status_report *report)
1523
1524{
1525	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1526	       core->revision == -1);
1527	return si476x_cmds_vtable[core->revision].fm_rsq_status(core, args,
1528								report);
1529}
1530EXPORT_SYMBOL_GPL(si476x_core_cmd_fm_rsq_status);
1531
1532int si476x_core_cmd_agc_status(struct si476x_core *core,
1533				  struct si476x_agc_status_report *report)
1534
1535{
1536	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1537	       core->revision == -1);
1538	return si476x_cmds_vtable[core->revision].agc_status(core, report);
1539}
1540EXPORT_SYMBOL_GPL(si476x_core_cmd_agc_status);
1541
1542int si476x_core_cmd_intb_pin_cfg(struct si476x_core *core,
1543			    enum si476x_intb_config intb,
1544			    enum si476x_a1_config a1)
1545{
1546	BUG_ON(core->revision > SI476X_REVISION_A30 ||
1547	       core->revision == -1);
1548
1549	return si476x_cmds_vtable[core->revision].intb_pin_cfg(core, intb, a1);
1550}
1551EXPORT_SYMBOL_GPL(si476x_core_cmd_intb_pin_cfg);
1552
1553MODULE_LICENSE("GPL");
1554MODULE_AUTHOR("Andrey Smirnov <andrew.smirnov@gmail.com>");
1555MODULE_DESCRIPTION("API for command exchange for si476x");
1556