1c942fddfSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
2b285192aSMauro Carvalho Chehab /*
3b285192aSMauro Carvalho Chehab  *  Driver for the NXP SAA7164 PCIe bridge
4b285192aSMauro Carvalho Chehab  *
563a412ecSSteven Toth  *  Copyright (c) 2010-2015 Steven Toth <stoth@kernellabs.com>
6b285192aSMauro Carvalho Chehab  */
7b285192aSMauro Carvalho Chehab 
8b285192aSMauro Carvalho Chehab #include <linux/wait.h>
9b285192aSMauro Carvalho Chehab #include <linux/slab.h>
10b285192aSMauro Carvalho Chehab 
11b285192aSMauro Carvalho Chehab #include "saa7164.h"
12b285192aSMauro Carvalho Chehab 
saa7164_api_get_load_info(struct saa7164_dev * dev,struct tmFwInfoStruct * i)13b285192aSMauro Carvalho Chehab int saa7164_api_get_load_info(struct saa7164_dev *dev, struct tmFwInfoStruct *i)
14b285192aSMauro Carvalho Chehab {
15b285192aSMauro Carvalho Chehab 	int ret;
16b285192aSMauro Carvalho Chehab 
17b285192aSMauro Carvalho Chehab 	if (!(saa_debug & DBGLVL_CPU))
18b285192aSMauro Carvalho Chehab 		return 0;
19b285192aSMauro Carvalho Chehab 
20b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s()\n", __func__);
21b285192aSMauro Carvalho Chehab 
22b285192aSMauro Carvalho Chehab 	i->deviceinst = 0;
23b285192aSMauro Carvalho Chehab 	i->devicespec = 0;
24b285192aSMauro Carvalho Chehab 	i->mode = 0;
25b285192aSMauro Carvalho Chehab 	i->status = 0;
26b285192aSMauro Carvalho Chehab 
27b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, 0, GET_CUR,
28b285192aSMauro Carvalho Chehab 		GET_FW_STATUS_CONTROL, sizeof(struct tmFwInfoStruct), i);
29b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
30b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
31b285192aSMauro Carvalho Chehab 
32b285192aSMauro Carvalho Chehab 	printk(KERN_INFO "saa7164[%d]-CPU: %d percent", dev->nr, i->CPULoad);
33b285192aSMauro Carvalho Chehab 
34b285192aSMauro Carvalho Chehab 	return ret;
35b285192aSMauro Carvalho Chehab }
36b285192aSMauro Carvalho Chehab 
saa7164_api_collect_debug(struct saa7164_dev * dev)37b285192aSMauro Carvalho Chehab int saa7164_api_collect_debug(struct saa7164_dev *dev)
38b285192aSMauro Carvalho Chehab {
39b285192aSMauro Carvalho Chehab 	struct tmComResDebugGetData d;
40b285192aSMauro Carvalho Chehab 	u8 more = 255;
41b285192aSMauro Carvalho Chehab 	int ret;
42b285192aSMauro Carvalho Chehab 
43b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s()\n", __func__);
44b285192aSMauro Carvalho Chehab 
45b285192aSMauro Carvalho Chehab 	while (more--) {
46b285192aSMauro Carvalho Chehab 
47b285192aSMauro Carvalho Chehab 		memset(&d, 0, sizeof(d));
48b285192aSMauro Carvalho Chehab 
49b285192aSMauro Carvalho Chehab 		ret = saa7164_cmd_send(dev, 0, GET_CUR,
50b285192aSMauro Carvalho Chehab 			GET_DEBUG_DATA_CONTROL, sizeof(d), &d);
51b285192aSMauro Carvalho Chehab 		if (ret != SAA_OK)
52b285192aSMauro Carvalho Chehab 			printk(KERN_ERR "%s() error, ret = 0x%x\n",
53b285192aSMauro Carvalho Chehab 				__func__, ret);
54b285192aSMauro Carvalho Chehab 
55b285192aSMauro Carvalho Chehab 		if (d.dwResult != SAA_OK)
56b285192aSMauro Carvalho Chehab 			break;
57b285192aSMauro Carvalho Chehab 
58b285192aSMauro Carvalho Chehab 		printk(KERN_INFO "saa7164[%d]-FWMSG: %s", dev->nr,
59b285192aSMauro Carvalho Chehab 			d.ucDebugData);
60b285192aSMauro Carvalho Chehab 	}
61b285192aSMauro Carvalho Chehab 
62b285192aSMauro Carvalho Chehab 	return 0;
63b285192aSMauro Carvalho Chehab }
64b285192aSMauro Carvalho Chehab 
saa7164_api_set_debug(struct saa7164_dev * dev,u8 level)65b285192aSMauro Carvalho Chehab int saa7164_api_set_debug(struct saa7164_dev *dev, u8 level)
66b285192aSMauro Carvalho Chehab {
67b285192aSMauro Carvalho Chehab 	struct tmComResDebugSetLevel lvl;
68b285192aSMauro Carvalho Chehab 	int ret;
69b285192aSMauro Carvalho Chehab 
70b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(level=%d)\n", __func__, level);
71b285192aSMauro Carvalho Chehab 
72b285192aSMauro Carvalho Chehab 	/* Retrieve current state */
73b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, 0, GET_CUR,
74b285192aSMauro Carvalho Chehab 		SET_DEBUG_LEVEL_CONTROL, sizeof(lvl), &lvl);
75b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
76b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
77b285192aSMauro Carvalho Chehab 
78b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s() Was %d\n", __func__, lvl.dwDebugLevel);
79b285192aSMauro Carvalho Chehab 
80b285192aSMauro Carvalho Chehab 	lvl.dwDebugLevel = level;
81b285192aSMauro Carvalho Chehab 
82b285192aSMauro Carvalho Chehab 	/* set new state */
83b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, 0, SET_CUR,
84b285192aSMauro Carvalho Chehab 		SET_DEBUG_LEVEL_CONTROL, sizeof(lvl), &lvl);
85b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
86b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
87b285192aSMauro Carvalho Chehab 
88b285192aSMauro Carvalho Chehab 	return ret;
89b285192aSMauro Carvalho Chehab }
90b285192aSMauro Carvalho Chehab 
saa7164_api_set_vbi_format(struct saa7164_port * port)91b285192aSMauro Carvalho Chehab int saa7164_api_set_vbi_format(struct saa7164_port *port)
92b285192aSMauro Carvalho Chehab {
93b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
94b285192aSMauro Carvalho Chehab 	struct tmComResProbeCommit fmt, rsp;
95b285192aSMauro Carvalho Chehab 	int ret;
96b285192aSMauro Carvalho Chehab 
97b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(nr=%d, unitid=0x%x)\n", __func__,
98b285192aSMauro Carvalho Chehab 		port->nr, port->hwcfg.unitid);
99b285192aSMauro Carvalho Chehab 
100b285192aSMauro Carvalho Chehab 	fmt.bmHint = 0;
101b285192aSMauro Carvalho Chehab 	fmt.bFormatIndex = 1;
102b285192aSMauro Carvalho Chehab 	fmt.bFrameIndex = 1;
103b285192aSMauro Carvalho Chehab 
104b285192aSMauro Carvalho Chehab 	/* Probe, see if it can support this format */
105b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.unitid,
106b285192aSMauro Carvalho Chehab 		SET_CUR, SAA_PROBE_CONTROL, sizeof(fmt), &fmt);
107b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
108b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() set error, ret = 0x%x\n", __func__, ret);
109b285192aSMauro Carvalho Chehab 
110b285192aSMauro Carvalho Chehab 	/* See of the format change was successful */
111b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.unitid,
112b285192aSMauro Carvalho Chehab 		GET_CUR, SAA_PROBE_CONTROL, sizeof(rsp), &rsp);
113b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK) {
114b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() get error, ret = 0x%x\n", __func__, ret);
115b285192aSMauro Carvalho Chehab 	} else {
116b285192aSMauro Carvalho Chehab 		/* Compare requested vs received, should be same */
117b285192aSMauro Carvalho Chehab 		if (memcmp(&fmt, &rsp, sizeof(rsp)) == 0) {
118b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "SET/PROBE Verified\n");
119b285192aSMauro Carvalho Chehab 
120b285192aSMauro Carvalho Chehab 			/* Ask the device to select the negotiated format */
121b285192aSMauro Carvalho Chehab 			ret = saa7164_cmd_send(port->dev, port->hwcfg.unitid,
122b285192aSMauro Carvalho Chehab 				SET_CUR, SAA_COMMIT_CONTROL, sizeof(fmt), &fmt);
123b285192aSMauro Carvalho Chehab 			if (ret != SAA_OK)
124b285192aSMauro Carvalho Chehab 				printk(KERN_ERR "%s() commit error, ret = 0x%x\n",
125b285192aSMauro Carvalho Chehab 					__func__, ret);
126b285192aSMauro Carvalho Chehab 
127b285192aSMauro Carvalho Chehab 			ret = saa7164_cmd_send(port->dev, port->hwcfg.unitid,
128b285192aSMauro Carvalho Chehab 				GET_CUR, SAA_COMMIT_CONTROL, sizeof(rsp), &rsp);
129b285192aSMauro Carvalho Chehab 			if (ret != SAA_OK)
130b285192aSMauro Carvalho Chehab 				printk(KERN_ERR "%s() GET commit error, ret = 0x%x\n",
131b285192aSMauro Carvalho Chehab 					__func__, ret);
132b285192aSMauro Carvalho Chehab 
133b285192aSMauro Carvalho Chehab 			if (memcmp(&fmt, &rsp, sizeof(rsp)) != 0) {
134b285192aSMauro Carvalho Chehab 				printk(KERN_ERR "%s() memcmp error, ret = 0x%x\n",
135b285192aSMauro Carvalho Chehab 					__func__, ret);
136b285192aSMauro Carvalho Chehab 			} else
137b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "SET/COMMIT Verified\n");
138b285192aSMauro Carvalho Chehab 
139b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "rsp.bmHint = 0x%x\n", rsp.bmHint);
140b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "rsp.bFormatIndex = 0x%x\n",
141b285192aSMauro Carvalho Chehab 				rsp.bFormatIndex);
142b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "rsp.bFrameIndex = 0x%x\n",
143b285192aSMauro Carvalho Chehab 				rsp.bFrameIndex);
144b285192aSMauro Carvalho Chehab 		} else
145b285192aSMauro Carvalho Chehab 			printk(KERN_ERR "%s() compare failed\n", __func__);
146b285192aSMauro Carvalho Chehab 	}
147b285192aSMauro Carvalho Chehab 
148b285192aSMauro Carvalho Chehab 	if (ret == SAA_OK)
149b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, "%s(nr=%d) Success\n", __func__, port->nr);
150b285192aSMauro Carvalho Chehab 
151b285192aSMauro Carvalho Chehab 	return ret;
152b285192aSMauro Carvalho Chehab }
153b285192aSMauro Carvalho Chehab 
saa7164_api_set_gop_size(struct saa7164_port * port)1545faf7db8SMauro Carvalho Chehab static int saa7164_api_set_gop_size(struct saa7164_port *port)
155b285192aSMauro Carvalho Chehab {
156b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
157b285192aSMauro Carvalho Chehab 	struct tmComResEncVideoGopStructure gs;
158b285192aSMauro Carvalho Chehab 	int ret;
159b285192aSMauro Carvalho Chehab 
160b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s()\n", __func__);
161b285192aSMauro Carvalho Chehab 
162b285192aSMauro Carvalho Chehab 	gs.ucRefFrameDist = port->encoder_params.refdist;
163b285192aSMauro Carvalho Chehab 	gs.ucGOPSize = port->encoder_params.gop_size;
164b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, SET_CUR,
165b285192aSMauro Carvalho Chehab 		EU_VIDEO_GOP_STRUCTURE_CONTROL,
166b285192aSMauro Carvalho Chehab 		sizeof(gs), &gs);
167b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
168b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
169b285192aSMauro Carvalho Chehab 
170b285192aSMauro Carvalho Chehab 	return ret;
171b285192aSMauro Carvalho Chehab }
172b285192aSMauro Carvalho Chehab 
saa7164_api_set_encoder(struct saa7164_port * port)173b285192aSMauro Carvalho Chehab int saa7164_api_set_encoder(struct saa7164_port *port)
174b285192aSMauro Carvalho Chehab {
175b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
176b285192aSMauro Carvalho Chehab 	struct tmComResEncVideoBitRate vb;
177b285192aSMauro Carvalho Chehab 	struct tmComResEncAudioBitRate ab;
178b285192aSMauro Carvalho Chehab 	int ret;
179b285192aSMauro Carvalho Chehab 
180b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s() unitid=0x%x\n", __func__,
181b285192aSMauro Carvalho Chehab 		port->hwcfg.sourceid);
182b285192aSMauro Carvalho Chehab 
183b285192aSMauro Carvalho Chehab 	if (port->encoder_params.stream_type == V4L2_MPEG_STREAM_TYPE_MPEG2_PS)
184b285192aSMauro Carvalho Chehab 		port->encoder_profile = EU_PROFILE_PS_DVD;
185b285192aSMauro Carvalho Chehab 	else
186b285192aSMauro Carvalho Chehab 		port->encoder_profile = EU_PROFILE_TS_HQ;
187b285192aSMauro Carvalho Chehab 
188b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, SET_CUR,
189b285192aSMauro Carvalho Chehab 		EU_PROFILE_CONTROL, sizeof(u8), &port->encoder_profile);
190b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
191b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
192b285192aSMauro Carvalho Chehab 
193b285192aSMauro Carvalho Chehab 	/* Resolution */
194b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, SET_CUR,
195b285192aSMauro Carvalho Chehab 		EU_PROFILE_CONTROL, sizeof(u8), &port->encoder_profile);
196b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
197b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
198b285192aSMauro Carvalho Chehab 
199b285192aSMauro Carvalho Chehab 	/* Establish video bitrates */
200b285192aSMauro Carvalho Chehab 	if (port->encoder_params.bitrate_mode ==
201b285192aSMauro Carvalho Chehab 		V4L2_MPEG_VIDEO_BITRATE_MODE_CBR)
202b285192aSMauro Carvalho Chehab 		vb.ucVideoBitRateMode = EU_VIDEO_BIT_RATE_MODE_CONSTANT;
203b285192aSMauro Carvalho Chehab 	else
204b285192aSMauro Carvalho Chehab 		vb.ucVideoBitRateMode = EU_VIDEO_BIT_RATE_MODE_VARIABLE_PEAK;
205b285192aSMauro Carvalho Chehab 	vb.dwVideoBitRate = port->encoder_params.bitrate;
206b285192aSMauro Carvalho Chehab 	vb.dwVideoBitRatePeak = port->encoder_params.bitrate_peak;
207b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, SET_CUR,
208b285192aSMauro Carvalho Chehab 		EU_VIDEO_BIT_RATE_CONTROL,
209b285192aSMauro Carvalho Chehab 		sizeof(struct tmComResEncVideoBitRate),
210b285192aSMauro Carvalho Chehab 		&vb);
211b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
212b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
213b285192aSMauro Carvalho Chehab 
214b285192aSMauro Carvalho Chehab 	/* Establish audio bitrates */
215b285192aSMauro Carvalho Chehab 	ab.ucAudioBitRateMode = 0;
216b285192aSMauro Carvalho Chehab 	ab.dwAudioBitRate = 384000;
217b285192aSMauro Carvalho Chehab 	ab.dwAudioBitRatePeak = ab.dwAudioBitRate;
218b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, SET_CUR,
219b285192aSMauro Carvalho Chehab 		EU_AUDIO_BIT_RATE_CONTROL,
220b285192aSMauro Carvalho Chehab 		sizeof(struct tmComResEncAudioBitRate),
221b285192aSMauro Carvalho Chehab 		&ab);
222b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
223b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__,
224b285192aSMauro Carvalho Chehab 			ret);
225b285192aSMauro Carvalho Chehab 
226b285192aSMauro Carvalho Chehab 	saa7164_api_set_aspect_ratio(port);
227b285192aSMauro Carvalho Chehab 	saa7164_api_set_gop_size(port);
228b285192aSMauro Carvalho Chehab 
229b285192aSMauro Carvalho Chehab 	return ret;
230b285192aSMauro Carvalho Chehab }
231b285192aSMauro Carvalho Chehab 
saa7164_api_get_encoder(struct saa7164_port * port)232b285192aSMauro Carvalho Chehab int saa7164_api_get_encoder(struct saa7164_port *port)
233b285192aSMauro Carvalho Chehab {
234b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
235b285192aSMauro Carvalho Chehab 	struct tmComResEncVideoBitRate v;
236b285192aSMauro Carvalho Chehab 	struct tmComResEncAudioBitRate a;
237b285192aSMauro Carvalho Chehab 	struct tmComResEncVideoInputAspectRatio ar;
238b285192aSMauro Carvalho Chehab 	int ret;
239b285192aSMauro Carvalho Chehab 
240b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s() unitid=0x%x\n", __func__,
241b285192aSMauro Carvalho Chehab 		port->hwcfg.sourceid);
242b285192aSMauro Carvalho Chehab 
243b285192aSMauro Carvalho Chehab 	port->encoder_profile = 0;
244b285192aSMauro Carvalho Chehab 	port->video_format = 0;
245b285192aSMauro Carvalho Chehab 	port->video_resolution = 0;
246b285192aSMauro Carvalho Chehab 	port->audio_format = 0;
247b285192aSMauro Carvalho Chehab 
248b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
249b285192aSMauro Carvalho Chehab 		EU_PROFILE_CONTROL, sizeof(u8), &port->encoder_profile);
250b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
251b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
252b285192aSMauro Carvalho Chehab 
253b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
254b285192aSMauro Carvalho Chehab 		EU_VIDEO_RESOLUTION_CONTROL, sizeof(u8),
255b285192aSMauro Carvalho Chehab 		&port->video_resolution);
256b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
257b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
258b285192aSMauro Carvalho Chehab 
259b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
260b285192aSMauro Carvalho Chehab 		EU_VIDEO_FORMAT_CONTROL, sizeof(u8), &port->video_format);
261b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
262b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
263b285192aSMauro Carvalho Chehab 
264b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
265b285192aSMauro Carvalho Chehab 		EU_VIDEO_BIT_RATE_CONTROL, sizeof(v), &v);
266b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
267b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
268b285192aSMauro Carvalho Chehab 
269b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
270b285192aSMauro Carvalho Chehab 		EU_AUDIO_FORMAT_CONTROL, sizeof(u8), &port->audio_format);
271b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
272b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
273b285192aSMauro Carvalho Chehab 
274b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
275b285192aSMauro Carvalho Chehab 		EU_AUDIO_BIT_RATE_CONTROL, sizeof(a), &a);
276b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
277b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
278b285192aSMauro Carvalho Chehab 
279b285192aSMauro Carvalho Chehab 	/* Aspect Ratio */
280b285192aSMauro Carvalho Chehab 	ar.width = 0;
281b285192aSMauro Carvalho Chehab 	ar.height = 0;
282b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, GET_CUR,
283b285192aSMauro Carvalho Chehab 		EU_VIDEO_INPUT_ASPECT_CONTROL,
284b285192aSMauro Carvalho Chehab 		sizeof(struct tmComResEncVideoInputAspectRatio), &ar);
285b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
286b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
287b285192aSMauro Carvalho Chehab 
288b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "encoder_profile = %d\n", port->encoder_profile);
289b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "video_format    = %d\n", port->video_format);
290b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "audio_format    = %d\n", port->audio_format);
291b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "video_resolution= %d\n", port->video_resolution);
292b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "v.ucVideoBitRateMode = %d\n",
293b285192aSMauro Carvalho Chehab 		v.ucVideoBitRateMode);
294b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "v.dwVideoBitRate     = %d\n",
295b285192aSMauro Carvalho Chehab 		v.dwVideoBitRate);
296b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "v.dwVideoBitRatePeak = %d\n",
297b285192aSMauro Carvalho Chehab 		v.dwVideoBitRatePeak);
298b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "a.ucVideoBitRateMode = %d\n",
299b285192aSMauro Carvalho Chehab 		a.ucAudioBitRateMode);
300b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "a.dwVideoBitRate     = %d\n",
301b285192aSMauro Carvalho Chehab 		a.dwAudioBitRate);
302b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "a.dwVideoBitRatePeak = %d\n",
303b285192aSMauro Carvalho Chehab 		a.dwAudioBitRatePeak);
304b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "aspect.width / height = %d:%d\n",
305b285192aSMauro Carvalho Chehab 		ar.width, ar.height);
306b285192aSMauro Carvalho Chehab 
307b285192aSMauro Carvalho Chehab 	return ret;
308b285192aSMauro Carvalho Chehab }
309b285192aSMauro Carvalho Chehab 
saa7164_api_set_aspect_ratio(struct saa7164_port * port)310b285192aSMauro Carvalho Chehab int saa7164_api_set_aspect_ratio(struct saa7164_port *port)
311b285192aSMauro Carvalho Chehab {
312b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
313b285192aSMauro Carvalho Chehab 	struct tmComResEncVideoInputAspectRatio ar;
314b285192aSMauro Carvalho Chehab 	int ret;
315b285192aSMauro Carvalho Chehab 
316b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s(%d)\n", __func__,
317b285192aSMauro Carvalho Chehab 		port->encoder_params.ctl_aspect);
318b285192aSMauro Carvalho Chehab 
319b285192aSMauro Carvalho Chehab 	switch (port->encoder_params.ctl_aspect) {
320b285192aSMauro Carvalho Chehab 	case V4L2_MPEG_VIDEO_ASPECT_1x1:
321b285192aSMauro Carvalho Chehab 		ar.width = 1;
322b285192aSMauro Carvalho Chehab 		ar.height = 1;
323b285192aSMauro Carvalho Chehab 		break;
324b285192aSMauro Carvalho Chehab 	case V4L2_MPEG_VIDEO_ASPECT_4x3:
325b285192aSMauro Carvalho Chehab 		ar.width = 4;
326b285192aSMauro Carvalho Chehab 		ar.height = 3;
327b285192aSMauro Carvalho Chehab 		break;
328b285192aSMauro Carvalho Chehab 	case V4L2_MPEG_VIDEO_ASPECT_16x9:
329b285192aSMauro Carvalho Chehab 		ar.width = 16;
330b285192aSMauro Carvalho Chehab 		ar.height = 9;
331b285192aSMauro Carvalho Chehab 		break;
332b285192aSMauro Carvalho Chehab 	case V4L2_MPEG_VIDEO_ASPECT_221x100:
333b285192aSMauro Carvalho Chehab 		ar.width = 221;
334b285192aSMauro Carvalho Chehab 		ar.height = 100;
335b285192aSMauro Carvalho Chehab 		break;
336b285192aSMauro Carvalho Chehab 	default:
337b285192aSMauro Carvalho Chehab 		BUG();
338b285192aSMauro Carvalho Chehab 	}
339b285192aSMauro Carvalho Chehab 
340b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s(%d) now %d:%d\n", __func__,
341b285192aSMauro Carvalho Chehab 		port->encoder_params.ctl_aspect,
342b285192aSMauro Carvalho Chehab 		ar.width, ar.height);
343b285192aSMauro Carvalho Chehab 
344b285192aSMauro Carvalho Chehab 	/* Aspect Ratio */
345b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.sourceid, SET_CUR,
346b285192aSMauro Carvalho Chehab 		EU_VIDEO_INPUT_ASPECT_CONTROL,
347b285192aSMauro Carvalho Chehab 		sizeof(struct tmComResEncVideoInputAspectRatio), &ar);
348b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
349b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
350b285192aSMauro Carvalho Chehab 
351b285192aSMauro Carvalho Chehab 	return ret;
352b285192aSMauro Carvalho Chehab }
353b285192aSMauro Carvalho Chehab 
saa7164_api_set_usercontrol(struct saa7164_port * port,u8 ctl)354b285192aSMauro Carvalho Chehab int saa7164_api_set_usercontrol(struct saa7164_port *port, u8 ctl)
355b285192aSMauro Carvalho Chehab {
356b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
357b285192aSMauro Carvalho Chehab 	int ret;
358b285192aSMauro Carvalho Chehab 	u16 val;
359b285192aSMauro Carvalho Chehab 
360b285192aSMauro Carvalho Chehab 	if (ctl == PU_BRIGHTNESS_CONTROL)
361b285192aSMauro Carvalho Chehab 		val = port->ctl_brightness;
362b285192aSMauro Carvalho Chehab 	else
363b285192aSMauro Carvalho Chehab 	if (ctl == PU_CONTRAST_CONTROL)
364b285192aSMauro Carvalho Chehab 		val = port->ctl_contrast;
365b285192aSMauro Carvalho Chehab 	else
366b285192aSMauro Carvalho Chehab 	if (ctl == PU_HUE_CONTROL)
367b285192aSMauro Carvalho Chehab 		val = port->ctl_hue;
368b285192aSMauro Carvalho Chehab 	else
369b285192aSMauro Carvalho Chehab 	if (ctl == PU_SATURATION_CONTROL)
370b285192aSMauro Carvalho Chehab 		val = port->ctl_saturation;
371b285192aSMauro Carvalho Chehab 	else
372b285192aSMauro Carvalho Chehab 	if (ctl == PU_SHARPNESS_CONTROL)
373b285192aSMauro Carvalho Chehab 		val = port->ctl_sharpness;
374b285192aSMauro Carvalho Chehab 	else
375b285192aSMauro Carvalho Chehab 		return -EINVAL;
376b285192aSMauro Carvalho Chehab 
377b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s() unitid=0x%x ctl=%d, val=%d\n",
378b285192aSMauro Carvalho Chehab 		__func__, port->encunit.vsourceid, ctl, val);
379b285192aSMauro Carvalho Chehab 
380b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->encunit.vsourceid, SET_CUR,
381b285192aSMauro Carvalho Chehab 		ctl, sizeof(u16), &val);
382b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
383b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
384b285192aSMauro Carvalho Chehab 
385b285192aSMauro Carvalho Chehab 	return ret;
386b285192aSMauro Carvalho Chehab }
387b285192aSMauro Carvalho Chehab 
saa7164_api_get_usercontrol(struct saa7164_port * port,u8 ctl)388b285192aSMauro Carvalho Chehab int saa7164_api_get_usercontrol(struct saa7164_port *port, u8 ctl)
389b285192aSMauro Carvalho Chehab {
390b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
391b285192aSMauro Carvalho Chehab 	int ret;
392b285192aSMauro Carvalho Chehab 	u16 val;
393b285192aSMauro Carvalho Chehab 
394b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->encunit.vsourceid, GET_CUR,
395b285192aSMauro Carvalho Chehab 		ctl, sizeof(u16), &val);
396b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK) {
397b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
398b285192aSMauro Carvalho Chehab 		return ret;
399b285192aSMauro Carvalho Chehab 	}
400b285192aSMauro Carvalho Chehab 
401b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s() ctl=%d, val=%d\n",
402b285192aSMauro Carvalho Chehab 		__func__, ctl, val);
403b285192aSMauro Carvalho Chehab 
404b285192aSMauro Carvalho Chehab 	if (ctl == PU_BRIGHTNESS_CONTROL)
405b285192aSMauro Carvalho Chehab 		port->ctl_brightness = val;
406b285192aSMauro Carvalho Chehab 	else
407b285192aSMauro Carvalho Chehab 	if (ctl == PU_CONTRAST_CONTROL)
408b285192aSMauro Carvalho Chehab 		port->ctl_contrast = val;
409b285192aSMauro Carvalho Chehab 	else
410b285192aSMauro Carvalho Chehab 	if (ctl == PU_HUE_CONTROL)
411b285192aSMauro Carvalho Chehab 		port->ctl_hue = val;
412b285192aSMauro Carvalho Chehab 	else
413b285192aSMauro Carvalho Chehab 	if (ctl == PU_SATURATION_CONTROL)
414b285192aSMauro Carvalho Chehab 		port->ctl_saturation = val;
415b285192aSMauro Carvalho Chehab 	else
416b285192aSMauro Carvalho Chehab 	if (ctl == PU_SHARPNESS_CONTROL)
417b285192aSMauro Carvalho Chehab 		port->ctl_sharpness = val;
418b285192aSMauro Carvalho Chehab 
419b285192aSMauro Carvalho Chehab 	return ret;
420b285192aSMauro Carvalho Chehab }
421b285192aSMauro Carvalho Chehab 
saa7164_api_set_videomux(struct saa7164_port * port)422b285192aSMauro Carvalho Chehab int saa7164_api_set_videomux(struct saa7164_port *port)
423b285192aSMauro Carvalho Chehab {
424b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
425b285192aSMauro Carvalho Chehab 	u8 inputs[] = { 1, 2, 2, 2, 5, 5, 5 };
426b285192aSMauro Carvalho Chehab 	int ret;
427b285192aSMauro Carvalho Chehab 
428b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s() v_mux=%d a_mux=%d\n",
429b285192aSMauro Carvalho Chehab 		__func__, port->mux_input, inputs[port->mux_input - 1]);
430b285192aSMauro Carvalho Chehab 
431b285192aSMauro Carvalho Chehab 	/* Audio Mute */
432b285192aSMauro Carvalho Chehab 	ret = saa7164_api_audio_mute(port, 1);
433b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
434b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
435b285192aSMauro Carvalho Chehab 
436b285192aSMauro Carvalho Chehab 	/* Video Mux */
437b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->vidproc.sourceid, SET_CUR,
438b285192aSMauro Carvalho Chehab 		SU_INPUT_SELECT_CONTROL, sizeof(u8), &port->mux_input);
439b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
440b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
441b285192aSMauro Carvalho Chehab 
442b285192aSMauro Carvalho Chehab 	/* Audio Mux */
443b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.sourceid, SET_CUR,
444b285192aSMauro Carvalho Chehab 		SU_INPUT_SELECT_CONTROL, sizeof(u8),
445b285192aSMauro Carvalho Chehab 		&inputs[port->mux_input - 1]);
446b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
447b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
448b285192aSMauro Carvalho Chehab 
449b285192aSMauro Carvalho Chehab 	/* Audio UnMute */
450b285192aSMauro Carvalho Chehab 	ret = saa7164_api_audio_mute(port, 0);
451b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
452b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
453b285192aSMauro Carvalho Chehab 
454b285192aSMauro Carvalho Chehab 	return ret;
455b285192aSMauro Carvalho Chehab }
456b285192aSMauro Carvalho Chehab 
saa7164_api_audio_mute(struct saa7164_port * port,int mute)457b285192aSMauro Carvalho Chehab int saa7164_api_audio_mute(struct saa7164_port *port, int mute)
458b285192aSMauro Carvalho Chehab {
459b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
460b285192aSMauro Carvalho Chehab 	u8 v = mute;
461b285192aSMauro Carvalho Chehab 	int ret;
462b285192aSMauro Carvalho Chehab 
463b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(%d)\n", __func__, mute);
464b285192aSMauro Carvalho Chehab 
465b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, SET_CUR,
466b285192aSMauro Carvalho Chehab 		MUTE_CONTROL, sizeof(u8), &v);
467b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
468b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
469b285192aSMauro Carvalho Chehab 
470b285192aSMauro Carvalho Chehab 	return ret;
471b285192aSMauro Carvalho Chehab }
472b285192aSMauro Carvalho Chehab 
473b285192aSMauro Carvalho Chehab /* 0 = silence, 0xff = full */
saa7164_api_set_audio_volume(struct saa7164_port * port,s8 level)474b285192aSMauro Carvalho Chehab int saa7164_api_set_audio_volume(struct saa7164_port *port, s8 level)
475b285192aSMauro Carvalho Chehab {
476b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
477b285192aSMauro Carvalho Chehab 	s16 v, min, max;
478b285192aSMauro Carvalho Chehab 	int ret;
479b285192aSMauro Carvalho Chehab 
480b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(%d)\n", __func__, level);
481b285192aSMauro Carvalho Chehab 
482b285192aSMauro Carvalho Chehab 	/* Obtain the min/max ranges */
483b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, GET_MIN,
484b285192aSMauro Carvalho Chehab 		VOLUME_CONTROL, sizeof(u16), &min);
485b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
486b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
487b285192aSMauro Carvalho Chehab 
488b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, GET_MAX,
489b285192aSMauro Carvalho Chehab 		VOLUME_CONTROL, sizeof(u16), &max);
490b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
491b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
492b285192aSMauro Carvalho Chehab 
493b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, GET_CUR,
494b285192aSMauro Carvalho Chehab 		(0x01 << 8) | VOLUME_CONTROL, sizeof(u16), &v);
495b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
496b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
497b285192aSMauro Carvalho Chehab 
498b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(%d) min=%d max=%d cur=%d\n", __func__,
499b285192aSMauro Carvalho Chehab 		level, min, max, v);
500b285192aSMauro Carvalho Chehab 
501b285192aSMauro Carvalho Chehab 	v = level;
502b285192aSMauro Carvalho Chehab 	if (v < min)
503b285192aSMauro Carvalho Chehab 		v = min;
504b285192aSMauro Carvalho Chehab 	if (v > max)
505b285192aSMauro Carvalho Chehab 		v = max;
506b285192aSMauro Carvalho Chehab 
507b285192aSMauro Carvalho Chehab 	/* Left */
508b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, SET_CUR,
509b285192aSMauro Carvalho Chehab 		(0x01 << 8) | VOLUME_CONTROL, sizeof(s16), &v);
510b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
511b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
512b285192aSMauro Carvalho Chehab 
513b285192aSMauro Carvalho Chehab 	/* Right */
514b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, SET_CUR,
515b285192aSMauro Carvalho Chehab 		(0x02 << 8) | VOLUME_CONTROL, sizeof(s16), &v);
516b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
517b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
518b285192aSMauro Carvalho Chehab 
519b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, GET_CUR,
520b285192aSMauro Carvalho Chehab 		(0x01 << 8) | VOLUME_CONTROL, sizeof(u16), &v);
521b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
522b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
523b285192aSMauro Carvalho Chehab 
524b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(%d) min=%d max=%d cur=%d\n", __func__,
525b285192aSMauro Carvalho Chehab 		level, min, max, v);
526b285192aSMauro Carvalho Chehab 
527b285192aSMauro Carvalho Chehab 	return ret;
528b285192aSMauro Carvalho Chehab }
529b285192aSMauro Carvalho Chehab 
saa7164_api_set_audio_std(struct saa7164_port * port)530b285192aSMauro Carvalho Chehab int saa7164_api_set_audio_std(struct saa7164_port *port)
531b285192aSMauro Carvalho Chehab {
532b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
533b285192aSMauro Carvalho Chehab 	struct tmComResAudioDefaults lvl;
534b285192aSMauro Carvalho Chehab 	struct tmComResTunerStandard tvaudio;
535b285192aSMauro Carvalho Chehab 	int ret;
536b285192aSMauro Carvalho Chehab 
537b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s()\n", __func__);
538b285192aSMauro Carvalho Chehab 
539b285192aSMauro Carvalho Chehab 	/* Establish default levels */
540b285192aSMauro Carvalho Chehab 	lvl.ucDecoderLevel = TMHW_LEV_ADJ_DECLEV_DEFAULT;
541b285192aSMauro Carvalho Chehab 	lvl.ucDecoderFM_Level = TMHW_LEV_ADJ_DECLEV_DEFAULT;
542b285192aSMauro Carvalho Chehab 	lvl.ucMonoLevel = TMHW_LEV_ADJ_MONOLEV_DEFAULT;
543b285192aSMauro Carvalho Chehab 	lvl.ucNICAM_Level = TMHW_LEV_ADJ_NICLEV_DEFAULT;
544b285192aSMauro Carvalho Chehab 	lvl.ucSAP_Level = TMHW_LEV_ADJ_SAPLEV_DEFAULT;
545b285192aSMauro Carvalho Chehab 	lvl.ucADC_Level = TMHW_LEV_ADJ_ADCLEV_DEFAULT;
546b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->audfeat.unitid, SET_CUR,
547b285192aSMauro Carvalho Chehab 		AUDIO_DEFAULT_CONTROL, sizeof(struct tmComResAudioDefaults),
548b285192aSMauro Carvalho Chehab 		&lvl);
549b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
550b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
551b285192aSMauro Carvalho Chehab 
552b285192aSMauro Carvalho Chehab 	/* Manually select the appropriate TV audio standard */
553b285192aSMauro Carvalho Chehab 	if (port->encodernorm.id & V4L2_STD_NTSC) {
554b285192aSMauro Carvalho Chehab 		tvaudio.std = TU_STANDARD_NTSC_M;
555b285192aSMauro Carvalho Chehab 		tvaudio.country = 1;
556b285192aSMauro Carvalho Chehab 	} else {
557b285192aSMauro Carvalho Chehab 		tvaudio.std = TU_STANDARD_PAL_I;
558b285192aSMauro Carvalho Chehab 		tvaudio.country = 44;
559b285192aSMauro Carvalho Chehab 	}
560b285192aSMauro Carvalho Chehab 
561b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->tunerunit.unitid, SET_CUR,
562b285192aSMauro Carvalho Chehab 		TU_STANDARD_CONTROL, sizeof(tvaudio), &tvaudio);
563b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
564b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() TU_STANDARD_CONTROL error, ret = 0x%x\n",
565b285192aSMauro Carvalho Chehab 			__func__, ret);
566b285192aSMauro Carvalho Chehab 	return ret;
567b285192aSMauro Carvalho Chehab }
568b285192aSMauro Carvalho Chehab 
saa7164_api_set_audio_detection(struct saa7164_port * port,int autodetect)569b285192aSMauro Carvalho Chehab int saa7164_api_set_audio_detection(struct saa7164_port *port, int autodetect)
570b285192aSMauro Carvalho Chehab {
571b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
572b285192aSMauro Carvalho Chehab 	struct tmComResTunerStandardAuto p;
573b285192aSMauro Carvalho Chehab 	int ret;
574b285192aSMauro Carvalho Chehab 
575b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(%d)\n", __func__, autodetect);
576b285192aSMauro Carvalho Chehab 
577b285192aSMauro Carvalho Chehab 	/* Disable TV Audio autodetect if not already set (buggy) */
578b285192aSMauro Carvalho Chehab 	if (autodetect)
579b285192aSMauro Carvalho Chehab 		p.mode = TU_STANDARD_AUTO;
580b285192aSMauro Carvalho Chehab 	else
581b285192aSMauro Carvalho Chehab 		p.mode = TU_STANDARD_MANUAL;
582b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->tunerunit.unitid, SET_CUR,
583b285192aSMauro Carvalho Chehab 		TU_STANDARD_AUTO_CONTROL, sizeof(p), &p);
584b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
585b285192aSMauro Carvalho Chehab 		printk(KERN_ERR
586b285192aSMauro Carvalho Chehab 			"%s() TU_STANDARD_AUTO_CONTROL error, ret = 0x%x\n",
587b285192aSMauro Carvalho Chehab 			__func__, ret);
588b285192aSMauro Carvalho Chehab 
589b285192aSMauro Carvalho Chehab 	return ret;
590b285192aSMauro Carvalho Chehab }
591b285192aSMauro Carvalho Chehab 
saa7164_api_get_videomux(struct saa7164_port * port)592b285192aSMauro Carvalho Chehab int saa7164_api_get_videomux(struct saa7164_port *port)
593b285192aSMauro Carvalho Chehab {
594b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
595b285192aSMauro Carvalho Chehab 	int ret;
596b285192aSMauro Carvalho Chehab 
597b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->vidproc.sourceid, GET_CUR,
598b285192aSMauro Carvalho Chehab 		SU_INPUT_SELECT_CONTROL, sizeof(u8), &port->mux_input);
599b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
600b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
601b285192aSMauro Carvalho Chehab 
602b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_ENC, "%s() v_mux=%d\n",
603b285192aSMauro Carvalho Chehab 		__func__, port->mux_input);
604b285192aSMauro Carvalho Chehab 
605b285192aSMauro Carvalho Chehab 	return ret;
606b285192aSMauro Carvalho Chehab }
607b285192aSMauro Carvalho Chehab 
saa7164_api_set_dif(struct saa7164_port * port,u8 reg,u8 val)6085faf7db8SMauro Carvalho Chehab static int saa7164_api_set_dif(struct saa7164_port *port, u8 reg, u8 val)
609b285192aSMauro Carvalho Chehab {
610b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
611b285192aSMauro Carvalho Chehab 
612b285192aSMauro Carvalho Chehab 	u16 len = 0;
613b285192aSMauro Carvalho Chehab 	u8 buf[256];
614b285192aSMauro Carvalho Chehab 	int ret;
615b285192aSMauro Carvalho Chehab 	u8 mas;
616b285192aSMauro Carvalho Chehab 
617b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(nr=%d type=%d val=%x)\n", __func__,
618b285192aSMauro Carvalho Chehab 		port->nr, port->type, val);
619b285192aSMauro Carvalho Chehab 
620b285192aSMauro Carvalho Chehab 	if (port->nr == 0)
621b285192aSMauro Carvalho Chehab 		mas = 0xd0;
622b285192aSMauro Carvalho Chehab 	else
623b285192aSMauro Carvalho Chehab 		mas = 0xe0;
624b285192aSMauro Carvalho Chehab 
625b285192aSMauro Carvalho Chehab 	memset(buf, 0, sizeof(buf));
626b285192aSMauro Carvalho Chehab 
627b285192aSMauro Carvalho Chehab 	buf[0x00] = 0x04;
628b285192aSMauro Carvalho Chehab 	buf[0x01] = 0x00;
629b285192aSMauro Carvalho Chehab 	buf[0x02] = 0x00;
630b285192aSMauro Carvalho Chehab 	buf[0x03] = 0x00;
631b285192aSMauro Carvalho Chehab 
632b285192aSMauro Carvalho Chehab 	buf[0x04] = 0x04;
633b285192aSMauro Carvalho Chehab 	buf[0x05] = 0x00;
634b285192aSMauro Carvalho Chehab 	buf[0x06] = 0x00;
635b285192aSMauro Carvalho Chehab 	buf[0x07] = 0x00;
636b285192aSMauro Carvalho Chehab 
637b285192aSMauro Carvalho Chehab 	buf[0x08] = reg;
638b285192aSMauro Carvalho Chehab 	buf[0x09] = 0x26;
639b285192aSMauro Carvalho Chehab 	buf[0x0a] = mas;
640b285192aSMauro Carvalho Chehab 	buf[0x0b] = 0xb0;
641b285192aSMauro Carvalho Chehab 
642b285192aSMauro Carvalho Chehab 	buf[0x0c] = val;
643b285192aSMauro Carvalho Chehab 	buf[0x0d] = 0x00;
644b285192aSMauro Carvalho Chehab 	buf[0x0e] = 0x00;
645b285192aSMauro Carvalho Chehab 	buf[0x0f] = 0x00;
646b285192aSMauro Carvalho Chehab 
647b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, port->ifunit.unitid, GET_LEN,
648b285192aSMauro Carvalho Chehab 		EXU_REGISTER_ACCESS_CONTROL, sizeof(len), &len);
649b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK) {
650b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret(1) = 0x%x\n", __func__, ret);
651b285192aSMauro Carvalho Chehab 		return -EIO;
652b285192aSMauro Carvalho Chehab 	}
653b285192aSMauro Carvalho Chehab 
654b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, port->ifunit.unitid, SET_CUR,
655b285192aSMauro Carvalho Chehab 		EXU_REGISTER_ACCESS_CONTROL, len, &buf);
656b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
657b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret(2) = 0x%x\n", __func__, ret);
658b285192aSMauro Carvalho Chehab #if 0
659b285192aSMauro Carvalho Chehab 	print_hex_dump(KERN_INFO, "", DUMP_PREFIX_OFFSET, 16, 1, buf, 16,
660b285192aSMauro Carvalho Chehab 		       false);
661b285192aSMauro Carvalho Chehab #endif
662b285192aSMauro Carvalho Chehab 	return ret == SAA_OK ? 0 : -EIO;
663b285192aSMauro Carvalho Chehab }
664b285192aSMauro Carvalho Chehab 
665b285192aSMauro Carvalho Chehab /* Disable the IF block AGC controls */
saa7164_api_configure_dif(struct saa7164_port * port,u32 std)666b285192aSMauro Carvalho Chehab int saa7164_api_configure_dif(struct saa7164_port *port, u32 std)
667b285192aSMauro Carvalho Chehab {
668b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
669b285192aSMauro Carvalho Chehab 	u8 agc_disable;
670b285192aSMauro Carvalho Chehab 
671b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(nr=%d, 0x%x)\n", __func__, port->nr, std);
672b285192aSMauro Carvalho Chehab 
673b285192aSMauro Carvalho Chehab 	if (std & V4L2_STD_NTSC) {
674b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " NTSC\n");
675b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x01); /* Video Standard */
676b285192aSMauro Carvalho Chehab 		agc_disable = 0;
677b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_PAL_I) {
678b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " PAL-I\n");
679b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x08); /* Video Standard */
680b285192aSMauro Carvalho Chehab 		agc_disable = 0;
681b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_PAL_M) {
682b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " PAL-M\n");
683b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x01); /* Video Standard */
684b285192aSMauro Carvalho Chehab 		agc_disable = 0;
685b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_PAL_N) {
686b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " PAL-N\n");
687b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x01); /* Video Standard */
688b285192aSMauro Carvalho Chehab 		agc_disable = 0;
689b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_PAL_Nc) {
690b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " PAL-Nc\n");
691b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x01); /* Video Standard */
692b285192aSMauro Carvalho Chehab 		agc_disable = 0;
693b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_PAL_B) {
694b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " PAL-B\n");
695b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x02); /* Video Standard */
696b285192aSMauro Carvalho Chehab 		agc_disable = 0;
697b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_PAL_DK) {
698b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " PAL-DK\n");
699b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x10); /* Video Standard */
700b285192aSMauro Carvalho Chehab 		agc_disable = 0;
701b285192aSMauro Carvalho Chehab 	} else if (std & V4L2_STD_SECAM_L) {
702b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " SECAM-L\n");
703b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x20); /* Video Standard */
704b285192aSMauro Carvalho Chehab 		agc_disable = 0;
705b285192aSMauro Carvalho Chehab 	} else {
706b285192aSMauro Carvalho Chehab 		/* Unknown standard, assume DTV */
707b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " Unknown (assuming DTV)\n");
708b285192aSMauro Carvalho Chehab 		/* Undefinded Video Standard */
709b285192aSMauro Carvalho Chehab 		saa7164_api_set_dif(port, 0x00, 0x80);
710b285192aSMauro Carvalho Chehab 		agc_disable = 1;
711b285192aSMauro Carvalho Chehab 	}
712b285192aSMauro Carvalho Chehab 
713b285192aSMauro Carvalho Chehab 	saa7164_api_set_dif(port, 0x48, 0xa0); /* AGC Functions 1 */
714b285192aSMauro Carvalho Chehab 	saa7164_api_set_dif(port, 0xc0, agc_disable); /* AGC Output Disable */
715b285192aSMauro Carvalho Chehab 	saa7164_api_set_dif(port, 0x7c, 0x04); /* CVBS EQ */
716b285192aSMauro Carvalho Chehab 	saa7164_api_set_dif(port, 0x04, 0x01); /* Active */
717b285192aSMauro Carvalho Chehab 	msleep(100);
718b285192aSMauro Carvalho Chehab 	saa7164_api_set_dif(port, 0x04, 0x00); /* Active (again) */
719b285192aSMauro Carvalho Chehab 	msleep(100);
720b285192aSMauro Carvalho Chehab 
7215b0eb827SMauro Carvalho Chehab 	return 0;
722b285192aSMauro Carvalho Chehab }
723b285192aSMauro Carvalho Chehab 
724b285192aSMauro Carvalho Chehab /* Ensure the dif is in the correct state for the operating mode
725b285192aSMauro Carvalho Chehab  * (analog / dtv). We only configure the diff through the analog encoder
726b285192aSMauro Carvalho Chehab  * so when we're in digital mode we need to find the appropriate encoder
727b285192aSMauro Carvalho Chehab  * and use it to configure the DIF.
728b285192aSMauro Carvalho Chehab  */
saa7164_api_initialize_dif(struct saa7164_port * port)729b285192aSMauro Carvalho Chehab int saa7164_api_initialize_dif(struct saa7164_port *port)
730b285192aSMauro Carvalho Chehab {
731b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
732b285192aSMauro Carvalho Chehab 	struct saa7164_port *p = NULL;
733b285192aSMauro Carvalho Chehab 	int ret = -EINVAL;
734b285192aSMauro Carvalho Chehab 	u32 std = 0;
735b285192aSMauro Carvalho Chehab 
736b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(nr=%d type=%d)\n", __func__,
737b285192aSMauro Carvalho Chehab 		port->nr, port->type);
738b285192aSMauro Carvalho Chehab 
739b285192aSMauro Carvalho Chehab 	if (port->type == SAA7164_MPEG_ENCODER) {
740b285192aSMauro Carvalho Chehab 		/* Pick any analog standard to init the diff.
741b285192aSMauro Carvalho Chehab 		 * we'll come back during encoder_init'
74216790554SMauro Carvalho Chehab 		 * and set the correct standard if required.
743b285192aSMauro Carvalho Chehab 		 */
744b285192aSMauro Carvalho Chehab 		std = V4L2_STD_NTSC;
745b285192aSMauro Carvalho Chehab 	} else
746b285192aSMauro Carvalho Chehab 	if (port->type == SAA7164_MPEG_DVB) {
747b285192aSMauro Carvalho Chehab 		if (port->nr == SAA7164_PORT_TS1)
748b285192aSMauro Carvalho Chehab 			p = &dev->ports[SAA7164_PORT_ENC1];
749b285192aSMauro Carvalho Chehab 		else
750b285192aSMauro Carvalho Chehab 			p = &dev->ports[SAA7164_PORT_ENC2];
751b285192aSMauro Carvalho Chehab 	} else
752b285192aSMauro Carvalho Chehab 	if (port->type == SAA7164_MPEG_VBI) {
753b285192aSMauro Carvalho Chehab 		std = V4L2_STD_NTSC;
754b285192aSMauro Carvalho Chehab 		if (port->nr == SAA7164_PORT_VBI1)
755b285192aSMauro Carvalho Chehab 			p = &dev->ports[SAA7164_PORT_ENC1];
756b285192aSMauro Carvalho Chehab 		else
757b285192aSMauro Carvalho Chehab 			p = &dev->ports[SAA7164_PORT_ENC2];
758b285192aSMauro Carvalho Chehab 	} else
759b285192aSMauro Carvalho Chehab 		BUG();
760b285192aSMauro Carvalho Chehab 
761b285192aSMauro Carvalho Chehab 	if (p)
762b285192aSMauro Carvalho Chehab 		ret = saa7164_api_configure_dif(p, std);
763b285192aSMauro Carvalho Chehab 
764b285192aSMauro Carvalho Chehab 	return ret;
765b285192aSMauro Carvalho Chehab }
766b285192aSMauro Carvalho Chehab 
saa7164_api_transition_port(struct saa7164_port * port,u8 mode)767b285192aSMauro Carvalho Chehab int saa7164_api_transition_port(struct saa7164_port *port, u8 mode)
768b285192aSMauro Carvalho Chehab {
769b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = port->dev;
770b285192aSMauro Carvalho Chehab 
771b285192aSMauro Carvalho Chehab 	int ret;
772b285192aSMauro Carvalho Chehab 
773b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(nr=%d unitid=0x%x,%d)\n",
774b285192aSMauro Carvalho Chehab 		__func__, port->nr, port->hwcfg.unitid, mode);
775b285192aSMauro Carvalho Chehab 
776b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(port->dev, port->hwcfg.unitid, SET_CUR,
777b285192aSMauro Carvalho Chehab 		SAA_STATE_CONTROL, sizeof(mode), &mode);
778b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
779b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s(portnr %d unitid 0x%x) error, ret = 0x%x\n",
780b285192aSMauro Carvalho Chehab 			__func__, port->nr, port->hwcfg.unitid, ret);
781b285192aSMauro Carvalho Chehab 
782b285192aSMauro Carvalho Chehab 	return ret;
783b285192aSMauro Carvalho Chehab }
784b285192aSMauro Carvalho Chehab 
saa7164_api_get_fw_version(struct saa7164_dev * dev,u32 * version)785b285192aSMauro Carvalho Chehab int saa7164_api_get_fw_version(struct saa7164_dev *dev, u32 *version)
786b285192aSMauro Carvalho Chehab {
787b285192aSMauro Carvalho Chehab 	int ret;
788b285192aSMauro Carvalho Chehab 
789b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, 0, GET_CUR,
790b285192aSMauro Carvalho Chehab 		GET_FW_VERSION_CONTROL, sizeof(u32), version);
791b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
792b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
793b285192aSMauro Carvalho Chehab 
794b285192aSMauro Carvalho Chehab 	return ret;
795b285192aSMauro Carvalho Chehab }
796b285192aSMauro Carvalho Chehab 
saa7164_api_read_eeprom(struct saa7164_dev * dev,u8 * buf,int buflen)797b285192aSMauro Carvalho Chehab int saa7164_api_read_eeprom(struct saa7164_dev *dev, u8 *buf, int buflen)
798b285192aSMauro Carvalho Chehab {
799b285192aSMauro Carvalho Chehab 	u8 reg[] = { 0x0f, 0x00 };
800b285192aSMauro Carvalho Chehab 
801b285192aSMauro Carvalho Chehab 	if (buflen < 128)
802b285192aSMauro Carvalho Chehab 		return -ENOMEM;
803b285192aSMauro Carvalho Chehab 
804*4502aae0SJiang Jian 	/* Assumption: Hauppauge eeprom is at 0xa0 on bus 0 */
805b285192aSMauro Carvalho Chehab 	/* TODO: Pull the details from the boards struct */
806b285192aSMauro Carvalho Chehab 	return saa7164_api_i2c_read(&dev->i2c_bus[0], 0xa0 >> 1, sizeof(reg),
807b285192aSMauro Carvalho Chehab 		&reg[0], 128, buf);
808b285192aSMauro Carvalho Chehab }
809b285192aSMauro Carvalho Chehab 
saa7164_api_configure_port_vbi(struct saa7164_dev * dev,struct saa7164_port * port)8105faf7db8SMauro Carvalho Chehab static int saa7164_api_configure_port_vbi(struct saa7164_dev *dev,
811b285192aSMauro Carvalho Chehab 					  struct saa7164_port *port)
812b285192aSMauro Carvalho Chehab {
813b285192aSMauro Carvalho Chehab 	struct tmComResVBIFormatDescrHeader *fmt = &port->vbi_fmt_ntsc;
814b285192aSMauro Carvalho Chehab 
815b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bFormatIndex  = 0x%x\n", fmt->bFormatIndex);
816b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    VideoStandard = 0x%x\n", fmt->VideoStandard);
817b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    StartLine     = %d\n", fmt->StartLine);
818b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    EndLine       = %d\n", fmt->EndLine);
819b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    FieldRate     = %d\n", fmt->FieldRate);
820b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bNumLines     = %d\n", fmt->bNumLines);
821b285192aSMauro Carvalho Chehab 
822b285192aSMauro Carvalho Chehab 	/* Cache the hardware configuration in the port */
823b285192aSMauro Carvalho Chehab 
824b285192aSMauro Carvalho Chehab 	port->bufcounter = port->hwcfg.BARLocation;
825b285192aSMauro Carvalho Chehab 	port->pitch = port->hwcfg.BARLocation + (2 * sizeof(u32));
826b285192aSMauro Carvalho Chehab 	port->bufsize = port->hwcfg.BARLocation + (3 * sizeof(u32));
827b285192aSMauro Carvalho Chehab 	port->bufoffset = port->hwcfg.BARLocation + (4 * sizeof(u32));
828b285192aSMauro Carvalho Chehab 	port->bufptr32l = port->hwcfg.BARLocation +
829b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
830b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount) + sizeof(u32);
831b285192aSMauro Carvalho Chehab 	port->bufptr32h = port->hwcfg.BARLocation +
832b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
833b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount);
834b285192aSMauro Carvalho Chehab 	port->bufptr64 = port->hwcfg.BARLocation +
835b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
836b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount);
837b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "   = port->hwcfg.BARLocation = 0x%x\n",
838b285192aSMauro Carvalho Chehab 		port->hwcfg.BARLocation);
839b285192aSMauro Carvalho Chehab 
840b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "   = VS_FORMAT_VBI (becomes dev->en[%d])\n",
841b285192aSMauro Carvalho Chehab 		port->nr);
842b285192aSMauro Carvalho Chehab 
843b285192aSMauro Carvalho Chehab 	return 0;
844b285192aSMauro Carvalho Chehab }
845b285192aSMauro Carvalho Chehab 
8465faf7db8SMauro Carvalho Chehab static int
saa7164_api_configure_port_mpeg2ts(struct saa7164_dev * dev,struct saa7164_port * port,struct tmComResTSFormatDescrHeader * tsfmt)8475faf7db8SMauro Carvalho Chehab saa7164_api_configure_port_mpeg2ts(struct saa7164_dev *dev,
848b285192aSMauro Carvalho Chehab 				   struct saa7164_port *port,
849b285192aSMauro Carvalho Chehab 				   struct tmComResTSFormatDescrHeader *tsfmt)
850b285192aSMauro Carvalho Chehab {
851b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bFormatIndex = 0x%x\n", tsfmt->bFormatIndex);
852b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bDataOffset  = 0x%x\n", tsfmt->bDataOffset);
853b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bPacketLength= 0x%x\n", tsfmt->bPacketLength);
854b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bStrideLength= 0x%x\n", tsfmt->bStrideLength);
855b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bguid        = (....)\n");
856b285192aSMauro Carvalho Chehab 
857b285192aSMauro Carvalho Chehab 	/* Cache the hardware configuration in the port */
858b285192aSMauro Carvalho Chehab 
859b285192aSMauro Carvalho Chehab 	port->bufcounter = port->hwcfg.BARLocation;
860b285192aSMauro Carvalho Chehab 	port->pitch = port->hwcfg.BARLocation + (2 * sizeof(u32));
861b285192aSMauro Carvalho Chehab 	port->bufsize = port->hwcfg.BARLocation + (3 * sizeof(u32));
862b285192aSMauro Carvalho Chehab 	port->bufoffset = port->hwcfg.BARLocation + (4 * sizeof(u32));
863b285192aSMauro Carvalho Chehab 	port->bufptr32l = port->hwcfg.BARLocation +
864b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
865b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount) + sizeof(u32);
866b285192aSMauro Carvalho Chehab 	port->bufptr32h = port->hwcfg.BARLocation +
867b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
868b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount);
869b285192aSMauro Carvalho Chehab 	port->bufptr64 = port->hwcfg.BARLocation +
870b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
871b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount);
872b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "   = port->hwcfg.BARLocation = 0x%x\n",
873b285192aSMauro Carvalho Chehab 		port->hwcfg.BARLocation);
874b285192aSMauro Carvalho Chehab 
875b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "   = VS_FORMAT_MPEGTS (becomes dev->ts[%d])\n",
876b285192aSMauro Carvalho Chehab 		port->nr);
877b285192aSMauro Carvalho Chehab 
878b285192aSMauro Carvalho Chehab 	return 0;
879b285192aSMauro Carvalho Chehab }
880b285192aSMauro Carvalho Chehab 
8815faf7db8SMauro Carvalho Chehab static int
saa7164_api_configure_port_mpeg2ps(struct saa7164_dev * dev,struct saa7164_port * port,struct tmComResPSFormatDescrHeader * fmt)8825faf7db8SMauro Carvalho Chehab saa7164_api_configure_port_mpeg2ps(struct saa7164_dev *dev,
883b285192aSMauro Carvalho Chehab 				   struct saa7164_port *port,
884b285192aSMauro Carvalho Chehab 				   struct tmComResPSFormatDescrHeader *fmt)
885b285192aSMauro Carvalho Chehab {
886b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bFormatIndex = 0x%x\n", fmt->bFormatIndex);
887b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    wPacketLength= 0x%x\n", fmt->wPacketLength);
888b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    wPackLength=   0x%x\n", fmt->wPackLength);
889b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "    bPackDataType= 0x%x\n", fmt->bPackDataType);
890b285192aSMauro Carvalho Chehab 
891b285192aSMauro Carvalho Chehab 	/* Cache the hardware configuration in the port */
892b285192aSMauro Carvalho Chehab 	/* TODO: CHECK THIS in the port config */
893b285192aSMauro Carvalho Chehab 	port->bufcounter = port->hwcfg.BARLocation;
894b285192aSMauro Carvalho Chehab 	port->pitch = port->hwcfg.BARLocation + (2 * sizeof(u32));
895b285192aSMauro Carvalho Chehab 	port->bufsize = port->hwcfg.BARLocation + (3 * sizeof(u32));
896b285192aSMauro Carvalho Chehab 	port->bufoffset = port->hwcfg.BARLocation + (4 * sizeof(u32));
897b285192aSMauro Carvalho Chehab 	port->bufptr32l = port->hwcfg.BARLocation +
898b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
899b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount) + sizeof(u32);
900b285192aSMauro Carvalho Chehab 	port->bufptr32h = port->hwcfg.BARLocation +
901b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
902b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount);
903b285192aSMauro Carvalho Chehab 	port->bufptr64 = port->hwcfg.BARLocation +
904b285192aSMauro Carvalho Chehab 		(4 * sizeof(u32)) +
905b285192aSMauro Carvalho Chehab 		(sizeof(u32) * port->hwcfg.buffercount);
906b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "   = port->hwcfg.BARLocation = 0x%x\n",
907b285192aSMauro Carvalho Chehab 		port->hwcfg.BARLocation);
908b285192aSMauro Carvalho Chehab 
909b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "   = VS_FORMAT_MPEGPS (becomes dev->enc[%d])\n",
910b285192aSMauro Carvalho Chehab 		port->nr);
911b285192aSMauro Carvalho Chehab 
912b285192aSMauro Carvalho Chehab 	return 0;
913b285192aSMauro Carvalho Chehab }
914b285192aSMauro Carvalho Chehab 
saa7164_api_dump_subdevs(struct saa7164_dev * dev,u8 * buf,int len)9155faf7db8SMauro Carvalho Chehab static int saa7164_api_dump_subdevs(struct saa7164_dev *dev, u8 *buf, int len)
916b285192aSMauro Carvalho Chehab {
917b285192aSMauro Carvalho Chehab 	struct saa7164_port *tsport = NULL;
918b285192aSMauro Carvalho Chehab 	struct saa7164_port *encport = NULL;
919b285192aSMauro Carvalho Chehab 	struct saa7164_port *vbiport = NULL;
920b285192aSMauro Carvalho Chehab 	u32 idx, next_offset;
921b285192aSMauro Carvalho Chehab 	int i;
922b285192aSMauro Carvalho Chehab 	struct tmComResDescrHeader *hdr, *t;
923b285192aSMauro Carvalho Chehab 	struct tmComResExtDevDescrHeader *exthdr;
924b285192aSMauro Carvalho Chehab 	struct tmComResPathDescrHeader *pathhdr;
925b285192aSMauro Carvalho Chehab 	struct tmComResAntTermDescrHeader *anttermhdr;
926b285192aSMauro Carvalho Chehab 	struct tmComResTunerDescrHeader *tunerunithdr;
927b285192aSMauro Carvalho Chehab 	struct tmComResDMATermDescrHeader *vcoutputtermhdr;
928b285192aSMauro Carvalho Chehab 	struct tmComResTSFormatDescrHeader *tsfmt;
929b285192aSMauro Carvalho Chehab 	struct tmComResPSFormatDescrHeader *psfmt;
930b285192aSMauro Carvalho Chehab 	struct tmComResSelDescrHeader *psel;
931b285192aSMauro Carvalho Chehab 	struct tmComResProcDescrHeader *pdh;
932b285192aSMauro Carvalho Chehab 	struct tmComResAFeatureDescrHeader *afd;
933b285192aSMauro Carvalho Chehab 	struct tmComResEncoderDescrHeader *edh;
934b285192aSMauro Carvalho Chehab 	struct tmComResVBIFormatDescrHeader *vbifmt;
935b285192aSMauro Carvalho Chehab 	u32 currpath = 0;
936b285192aSMauro Carvalho Chehab 
937b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API,
938b285192aSMauro Carvalho Chehab 		"%s(?,?,%d) sizeof(struct tmComResDescrHeader) = %d bytes\n",
939b285192aSMauro Carvalho Chehab 		__func__, len, (u32)sizeof(struct tmComResDescrHeader));
940b285192aSMauro Carvalho Chehab 
941b285192aSMauro Carvalho Chehab 	for (idx = 0; idx < (len - sizeof(struct tmComResDescrHeader));) {
942b285192aSMauro Carvalho Chehab 
943b285192aSMauro Carvalho Chehab 		hdr = (struct tmComResDescrHeader *)(buf + idx);
944b285192aSMauro Carvalho Chehab 
945b285192aSMauro Carvalho Chehab 		if (hdr->type != CS_INTERFACE)
946b285192aSMauro Carvalho Chehab 			return SAA_ERR_NOT_SUPPORTED;
947b285192aSMauro Carvalho Chehab 
948b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, "@ 0x%x =\n", idx);
949b285192aSMauro Carvalho Chehab 		switch (hdr->subtype) {
950b285192aSMauro Carvalho Chehab 		case GENERAL_REQUEST:
951b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " GENERAL_REQUEST\n");
952b285192aSMauro Carvalho Chehab 			break;
953b285192aSMauro Carvalho Chehab 		case VC_TUNER_PATH:
954b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " VC_TUNER_PATH\n");
955b285192aSMauro Carvalho Chehab 			pathhdr = (struct tmComResPathDescrHeader *)(buf + idx);
956b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  pathid = 0x%x\n",
957b285192aSMauro Carvalho Chehab 				pathhdr->pathid);
958b285192aSMauro Carvalho Chehab 			currpath = pathhdr->pathid;
959b285192aSMauro Carvalho Chehab 			break;
960b285192aSMauro Carvalho Chehab 		case VC_INPUT_TERMINAL:
961b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " VC_INPUT_TERMINAL\n");
962b285192aSMauro Carvalho Chehab 			anttermhdr =
963b285192aSMauro Carvalho Chehab 				(struct tmComResAntTermDescrHeader *)(buf + idx);
964b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  terminalid   = 0x%x\n",
965b285192aSMauro Carvalho Chehab 				anttermhdr->terminalid);
966b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  terminaltype = 0x%x\n",
967b285192aSMauro Carvalho Chehab 				anttermhdr->terminaltype);
968b285192aSMauro Carvalho Chehab 			switch (anttermhdr->terminaltype) {
969b285192aSMauro Carvalho Chehab 			case ITT_ANTENNA:
970b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = ITT_ANTENNA\n");
971b285192aSMauro Carvalho Chehab 				break;
972b285192aSMauro Carvalho Chehab 			case LINE_CONNECTOR:
973b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = LINE_CONNECTOR\n");
974b285192aSMauro Carvalho Chehab 				break;
975b285192aSMauro Carvalho Chehab 			case SPDIF_CONNECTOR:
976b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = SPDIF_CONNECTOR\n");
977b285192aSMauro Carvalho Chehab 				break;
978b285192aSMauro Carvalho Chehab 			case COMPOSITE_CONNECTOR:
979b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
980b285192aSMauro Carvalho Chehab 					"   = COMPOSITE_CONNECTOR\n");
981b285192aSMauro Carvalho Chehab 				break;
982b285192aSMauro Carvalho Chehab 			case SVIDEO_CONNECTOR:
983b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = SVIDEO_CONNECTOR\n");
984b285192aSMauro Carvalho Chehab 				break;
985b285192aSMauro Carvalho Chehab 			case COMPONENT_CONNECTOR:
986b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
987b285192aSMauro Carvalho Chehab 					"   = COMPONENT_CONNECTOR\n");
988b285192aSMauro Carvalho Chehab 				break;
989b285192aSMauro Carvalho Chehab 			case STANDARD_DMA:
990b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = STANDARD_DMA\n");
991b285192aSMauro Carvalho Chehab 				break;
992b285192aSMauro Carvalho Chehab 			default:
993b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = undefined (0x%x)\n",
994b285192aSMauro Carvalho Chehab 					anttermhdr->terminaltype);
995b285192aSMauro Carvalho Chehab 			}
996b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  assocterminal= 0x%x\n",
997b285192aSMauro Carvalho Chehab 				anttermhdr->assocterminal);
998b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  iterminal    = 0x%x\n",
999b285192aSMauro Carvalho Chehab 				anttermhdr->iterminal);
1000b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controlsize  = 0x%x\n",
1001b285192aSMauro Carvalho Chehab 				anttermhdr->controlsize);
1002b285192aSMauro Carvalho Chehab 			break;
1003b285192aSMauro Carvalho Chehab 		case VC_OUTPUT_TERMINAL:
1004b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " VC_OUTPUT_TERMINAL\n");
1005b285192aSMauro Carvalho Chehab 			vcoutputtermhdr =
1006b285192aSMauro Carvalho Chehab 				(struct tmComResDMATermDescrHeader *)(buf + idx);
1007b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n",
1008b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->unitid);
1009b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  terminaltype = 0x%x\n",
1010b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->terminaltype);
1011b285192aSMauro Carvalho Chehab 			switch (vcoutputtermhdr->terminaltype) {
1012b285192aSMauro Carvalho Chehab 			case ITT_ANTENNA:
1013b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = ITT_ANTENNA\n");
1014b285192aSMauro Carvalho Chehab 				break;
1015b285192aSMauro Carvalho Chehab 			case LINE_CONNECTOR:
1016b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = LINE_CONNECTOR\n");
1017b285192aSMauro Carvalho Chehab 				break;
1018b285192aSMauro Carvalho Chehab 			case SPDIF_CONNECTOR:
1019b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = SPDIF_CONNECTOR\n");
1020b285192aSMauro Carvalho Chehab 				break;
1021b285192aSMauro Carvalho Chehab 			case COMPOSITE_CONNECTOR:
1022b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1023b285192aSMauro Carvalho Chehab 					"   = COMPOSITE_CONNECTOR\n");
1024b285192aSMauro Carvalho Chehab 				break;
1025b285192aSMauro Carvalho Chehab 			case SVIDEO_CONNECTOR:
1026b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = SVIDEO_CONNECTOR\n");
1027b285192aSMauro Carvalho Chehab 				break;
1028b285192aSMauro Carvalho Chehab 			case COMPONENT_CONNECTOR:
1029b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1030b285192aSMauro Carvalho Chehab 					"   = COMPONENT_CONNECTOR\n");
1031b285192aSMauro Carvalho Chehab 				break;
1032b285192aSMauro Carvalho Chehab 			case STANDARD_DMA:
1033b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = STANDARD_DMA\n");
1034b285192aSMauro Carvalho Chehab 				break;
1035b285192aSMauro Carvalho Chehab 			default:
1036b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = undefined (0x%x)\n",
1037b285192aSMauro Carvalho Chehab 					vcoutputtermhdr->terminaltype);
1038b285192aSMauro Carvalho Chehab 			}
1039b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  assocterminal= 0x%x\n",
1040b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->assocterminal);
1041b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  sourceid     = 0x%x\n",
1042b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->sourceid);
1043b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  iterminal    = 0x%x\n",
1044b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->iterminal);
1045b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  BARLocation  = 0x%x\n",
1046b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->BARLocation);
1047b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  flags        = 0x%x\n",
1048b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->flags);
1049b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  interruptid  = 0x%x\n",
1050b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->interruptid);
1051b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  buffercount  = 0x%x\n",
1052b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->buffercount);
1053b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  metadatasize = 0x%x\n",
1054b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->metadatasize);
1055b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controlsize  = 0x%x\n",
1056b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->controlsize);
1057b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  numformats   = 0x%x\n",
1058b285192aSMauro Carvalho Chehab 				vcoutputtermhdr->numformats);
1059b285192aSMauro Carvalho Chehab 
1060b285192aSMauro Carvalho Chehab 			next_offset = idx + (vcoutputtermhdr->len);
1061b285192aSMauro Carvalho Chehab 			for (i = 0; i < vcoutputtermhdr->numformats; i++) {
1062b285192aSMauro Carvalho Chehab 				t = (struct tmComResDescrHeader *)
1063b285192aSMauro Carvalho Chehab 					(buf + next_offset);
1064b285192aSMauro Carvalho Chehab 				switch (t->subtype) {
1065b285192aSMauro Carvalho Chehab 				case VS_FORMAT_MPEG2TS:
1066b285192aSMauro Carvalho Chehab 					tsfmt =
1067b285192aSMauro Carvalho Chehab 					(struct tmComResTSFormatDescrHeader *)t;
1068b285192aSMauro Carvalho Chehab 					if (currpath == 1)
1069b285192aSMauro Carvalho Chehab 						tsport = &dev->ports[SAA7164_PORT_TS1];
1070b285192aSMauro Carvalho Chehab 					else
1071b285192aSMauro Carvalho Chehab 						tsport = &dev->ports[SAA7164_PORT_TS2];
1072b285192aSMauro Carvalho Chehab 					memcpy(&tsport->hwcfg, vcoutputtermhdr,
1073b285192aSMauro Carvalho Chehab 						sizeof(*vcoutputtermhdr));
1074b285192aSMauro Carvalho Chehab 					saa7164_api_configure_port_mpeg2ts(dev,
1075b285192aSMauro Carvalho Chehab 						tsport, tsfmt);
1076b285192aSMauro Carvalho Chehab 					break;
1077b285192aSMauro Carvalho Chehab 				case VS_FORMAT_MPEG2PS:
1078b285192aSMauro Carvalho Chehab 					psfmt =
1079b285192aSMauro Carvalho Chehab 					(struct tmComResPSFormatDescrHeader *)t;
1080b285192aSMauro Carvalho Chehab 					if (currpath == 1)
1081b285192aSMauro Carvalho Chehab 						encport = &dev->ports[SAA7164_PORT_ENC1];
1082b285192aSMauro Carvalho Chehab 					else
1083b285192aSMauro Carvalho Chehab 						encport = &dev->ports[SAA7164_PORT_ENC2];
1084b285192aSMauro Carvalho Chehab 					memcpy(&encport->hwcfg, vcoutputtermhdr,
1085b285192aSMauro Carvalho Chehab 						sizeof(*vcoutputtermhdr));
1086b285192aSMauro Carvalho Chehab 					saa7164_api_configure_port_mpeg2ps(dev,
1087b285192aSMauro Carvalho Chehab 						encport, psfmt);
1088b285192aSMauro Carvalho Chehab 					break;
1089b285192aSMauro Carvalho Chehab 				case VS_FORMAT_VBI:
1090b285192aSMauro Carvalho Chehab 					vbifmt =
1091b285192aSMauro Carvalho Chehab 					(struct tmComResVBIFormatDescrHeader *)t;
1092b285192aSMauro Carvalho Chehab 					if (currpath == 1)
1093b285192aSMauro Carvalho Chehab 						vbiport = &dev->ports[SAA7164_PORT_VBI1];
1094b285192aSMauro Carvalho Chehab 					else
1095b285192aSMauro Carvalho Chehab 						vbiport = &dev->ports[SAA7164_PORT_VBI2];
1096b285192aSMauro Carvalho Chehab 					memcpy(&vbiport->hwcfg, vcoutputtermhdr,
1097b285192aSMauro Carvalho Chehab 						sizeof(*vcoutputtermhdr));
1098b285192aSMauro Carvalho Chehab 					memcpy(&vbiport->vbi_fmt_ntsc, vbifmt,
1099b285192aSMauro Carvalho Chehab 						sizeof(*vbifmt));
1100b285192aSMauro Carvalho Chehab 					saa7164_api_configure_port_vbi(dev,
1101b285192aSMauro Carvalho Chehab 						vbiport);
1102b285192aSMauro Carvalho Chehab 					break;
1103b285192aSMauro Carvalho Chehab 				case VS_FORMAT_RDS:
1104b285192aSMauro Carvalho Chehab 					dprintk(DBGLVL_API,
1105b285192aSMauro Carvalho Chehab 						"   = VS_FORMAT_RDS\n");
1106b285192aSMauro Carvalho Chehab 					break;
1107b285192aSMauro Carvalho Chehab 				case VS_FORMAT_UNCOMPRESSED:
1108b285192aSMauro Carvalho Chehab 					dprintk(DBGLVL_API,
1109b285192aSMauro Carvalho Chehab 					"   = VS_FORMAT_UNCOMPRESSED\n");
1110b285192aSMauro Carvalho Chehab 					break;
1111b285192aSMauro Carvalho Chehab 				case VS_FORMAT_TYPE:
1112b285192aSMauro Carvalho Chehab 					dprintk(DBGLVL_API,
1113b285192aSMauro Carvalho Chehab 						"   = VS_FORMAT_TYPE\n");
1114b285192aSMauro Carvalho Chehab 					break;
1115b285192aSMauro Carvalho Chehab 				default:
1116b285192aSMauro Carvalho Chehab 					dprintk(DBGLVL_API,
1117b285192aSMauro Carvalho Chehab 						"   = undefined (0x%x)\n",
1118b285192aSMauro Carvalho Chehab 						t->subtype);
1119b285192aSMauro Carvalho Chehab 				}
1120b285192aSMauro Carvalho Chehab 				next_offset += t->len;
1121b285192aSMauro Carvalho Chehab 			}
1122b285192aSMauro Carvalho Chehab 
1123b285192aSMauro Carvalho Chehab 			break;
1124b285192aSMauro Carvalho Chehab 		case TUNER_UNIT:
1125b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " TUNER_UNIT\n");
1126b285192aSMauro Carvalho Chehab 			tunerunithdr =
1127b285192aSMauro Carvalho Chehab 				(struct tmComResTunerDescrHeader *)(buf + idx);
1128b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n",
1129b285192aSMauro Carvalho Chehab 				tunerunithdr->unitid);
1130b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  sourceid = 0x%x\n",
1131b285192aSMauro Carvalho Chehab 				tunerunithdr->sourceid);
1132b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  iunit = 0x%x\n",
1133b285192aSMauro Carvalho Chehab 				tunerunithdr->iunit);
1134b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  tuningstandards = 0x%x\n",
1135b285192aSMauro Carvalho Chehab 				tunerunithdr->tuningstandards);
1136b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controlsize = 0x%x\n",
1137b285192aSMauro Carvalho Chehab 				tunerunithdr->controlsize);
1138b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controls = 0x%x\n",
1139b285192aSMauro Carvalho Chehab 				tunerunithdr->controls);
1140b285192aSMauro Carvalho Chehab 
1141b285192aSMauro Carvalho Chehab 			if (tunerunithdr->unitid == tunerunithdr->iunit) {
1142b285192aSMauro Carvalho Chehab 				if (currpath == 1)
1143b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC1];
1144b285192aSMauro Carvalho Chehab 				else
1145b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC2];
1146b285192aSMauro Carvalho Chehab 				memcpy(&encport->tunerunit, tunerunithdr,
1147b285192aSMauro Carvalho Chehab 					sizeof(struct tmComResTunerDescrHeader));
1148b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1149b285192aSMauro Carvalho Chehab 					"  (becomes dev->enc[%d] tuner)\n",
1150b285192aSMauro Carvalho Chehab 					encport->nr);
1151b285192aSMauro Carvalho Chehab 			}
1152b285192aSMauro Carvalho Chehab 			break;
1153b285192aSMauro Carvalho Chehab 		case VC_SELECTOR_UNIT:
1154b285192aSMauro Carvalho Chehab 			psel = (struct tmComResSelDescrHeader *)(buf + idx);
1155b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " VC_SELECTOR_UNIT\n");
1156b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n",
1157b285192aSMauro Carvalho Chehab 				psel->unitid);
1158b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  nrinpins = 0x%x\n",
1159b285192aSMauro Carvalho Chehab 				psel->nrinpins);
1160b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  sourceid = 0x%x\n",
1161b285192aSMauro Carvalho Chehab 				psel->sourceid);
1162b285192aSMauro Carvalho Chehab 			break;
1163b285192aSMauro Carvalho Chehab 		case VC_PROCESSING_UNIT:
1164b285192aSMauro Carvalho Chehab 			pdh = (struct tmComResProcDescrHeader *)(buf + idx);
1165b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " VC_PROCESSING_UNIT\n");
1166b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n",
1167b285192aSMauro Carvalho Chehab 				pdh->unitid);
1168b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  sourceid = 0x%x\n",
1169b285192aSMauro Carvalho Chehab 				pdh->sourceid);
1170b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controlsize = 0x%x\n",
1171b285192aSMauro Carvalho Chehab 				pdh->controlsize);
1172b285192aSMauro Carvalho Chehab 			if (pdh->controlsize == 0x04) {
1173b285192aSMauro Carvalho Chehab 				if (currpath == 1)
1174b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC1];
1175b285192aSMauro Carvalho Chehab 				else
1176b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC2];
1177b285192aSMauro Carvalho Chehab 				memcpy(&encport->vidproc, pdh,
1178b285192aSMauro Carvalho Chehab 					sizeof(struct tmComResProcDescrHeader));
1179b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "  (becomes dev->enc[%d])\n",
1180b285192aSMauro Carvalho Chehab 					encport->nr);
1181b285192aSMauro Carvalho Chehab 			}
1182b285192aSMauro Carvalho Chehab 			break;
1183b285192aSMauro Carvalho Chehab 		case FEATURE_UNIT:
1184b285192aSMauro Carvalho Chehab 			afd = (struct tmComResAFeatureDescrHeader *)(buf + idx);
1185b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " FEATURE_UNIT\n");
1186b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n",
1187b285192aSMauro Carvalho Chehab 				afd->unitid);
1188b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  sourceid = 0x%x\n",
1189b285192aSMauro Carvalho Chehab 				afd->sourceid);
1190b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controlsize = 0x%x\n",
1191b285192aSMauro Carvalho Chehab 				afd->controlsize);
1192b285192aSMauro Carvalho Chehab 			if (currpath == 1)
1193b285192aSMauro Carvalho Chehab 				encport = &dev->ports[SAA7164_PORT_ENC1];
1194b285192aSMauro Carvalho Chehab 			else
1195b285192aSMauro Carvalho Chehab 				encport = &dev->ports[SAA7164_PORT_ENC2];
1196b285192aSMauro Carvalho Chehab 			memcpy(&encport->audfeat, afd,
1197b285192aSMauro Carvalho Chehab 				sizeof(struct tmComResAFeatureDescrHeader));
1198b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  (becomes dev->enc[%d])\n",
1199b285192aSMauro Carvalho Chehab 				encport->nr);
1200b285192aSMauro Carvalho Chehab 			break;
1201b285192aSMauro Carvalho Chehab 		case ENCODER_UNIT:
1202b285192aSMauro Carvalho Chehab 			edh = (struct tmComResEncoderDescrHeader *)(buf + idx);
1203b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " ENCODER_UNIT\n");
1204b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  subtype = 0x%x\n", edh->subtype);
1205b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n", edh->unitid);
1206b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  vsourceid = 0x%x\n",
1207b285192aSMauro Carvalho Chehab 			edh->vsourceid);
1208b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  asourceid = 0x%x\n",
1209b285192aSMauro Carvalho Chehab 				edh->asourceid);
1210b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  iunit = 0x%x\n", edh->iunit);
1211b285192aSMauro Carvalho Chehab 			if (edh->iunit == edh->unitid) {
1212b285192aSMauro Carvalho Chehab 				if (currpath == 1)
1213b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC1];
1214b285192aSMauro Carvalho Chehab 				else
1215b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC2];
1216b285192aSMauro Carvalho Chehab 				memcpy(&encport->encunit, edh,
1217b285192aSMauro Carvalho Chehab 					sizeof(struct tmComResEncoderDescrHeader));
1218b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1219b285192aSMauro Carvalho Chehab 					"  (becomes dev->enc[%d])\n",
1220b285192aSMauro Carvalho Chehab 					encport->nr);
1221b285192aSMauro Carvalho Chehab 			}
1222b285192aSMauro Carvalho Chehab 			break;
1223b285192aSMauro Carvalho Chehab 		case EXTENSION_UNIT:
1224b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " EXTENSION_UNIT\n");
1225b285192aSMauro Carvalho Chehab 			exthdr = (struct tmComResExtDevDescrHeader *)(buf + idx);
1226b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  unitid = 0x%x\n",
1227b285192aSMauro Carvalho Chehab 				exthdr->unitid);
1228b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  deviceid = 0x%x\n",
1229b285192aSMauro Carvalho Chehab 				exthdr->deviceid);
1230b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  devicetype = 0x%x\n",
1231b285192aSMauro Carvalho Chehab 				exthdr->devicetype);
1232b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x1)
1233b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Decoder Device\n");
1234b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x2)
1235b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = GPIO Source\n");
1236b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x4)
1237b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Video Decoder\n");
1238b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x8)
1239b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Audio Decoder\n");
1240b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x20)
1241b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Crossbar\n");
1242b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x40)
1243b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Tuner\n");
1244b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x80)
1245b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = IF PLL\n");
1246b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x100)
1247b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Demodulator\n");
1248b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x200)
1249b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = RDS Decoder\n");
1250b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x400)
1251b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = Encoder\n");
1252b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x800)
1253b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = IR Decoder\n");
1254b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x1000)
1255b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API, "   = EEPROM\n");
1256b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x2000)
1257b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1258b285192aSMauro Carvalho Chehab 					"   = VBI Decoder\n");
1259b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x10000)
1260b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1261b285192aSMauro Carvalho Chehab 					"   = Streaming Device\n");
1262b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x20000)
1263b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1264b285192aSMauro Carvalho Chehab 					"   = DRM Device\n");
1265b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x40000000)
1266b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1267b285192aSMauro Carvalho Chehab 					"   = Generic Device\n");
1268b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x80000000)
1269b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1270b285192aSMauro Carvalho Chehab 					"   = Config Space Device\n");
1271b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  numgpiopins = 0x%x\n",
1272b285192aSMauro Carvalho Chehab 				exthdr->numgpiopins);
1273b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  numgpiogroups = 0x%x\n",
1274b285192aSMauro Carvalho Chehab 				exthdr->numgpiogroups);
1275b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "  controlsize = 0x%x\n",
1276b285192aSMauro Carvalho Chehab 				exthdr->controlsize);
1277b285192aSMauro Carvalho Chehab 			if (exthdr->devicetype & 0x80) {
1278b285192aSMauro Carvalho Chehab 				if (currpath == 1)
1279b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC1];
1280b285192aSMauro Carvalho Chehab 				else
1281b285192aSMauro Carvalho Chehab 					encport = &dev->ports[SAA7164_PORT_ENC2];
1282b285192aSMauro Carvalho Chehab 				memcpy(&encport->ifunit, exthdr,
1283b285192aSMauro Carvalho Chehab 					sizeof(struct tmComResExtDevDescrHeader));
1284b285192aSMauro Carvalho Chehab 				dprintk(DBGLVL_API,
1285b285192aSMauro Carvalho Chehab 					"  (becomes dev->enc[%d])\n",
1286b285192aSMauro Carvalho Chehab 					encport->nr);
1287b285192aSMauro Carvalho Chehab 			}
1288b285192aSMauro Carvalho Chehab 			break;
1289b285192aSMauro Carvalho Chehab 		case PVC_INFRARED_UNIT:
1290b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " PVC_INFRARED_UNIT\n");
1291b285192aSMauro Carvalho Chehab 			break;
1292b285192aSMauro Carvalho Chehab 		case DRM_UNIT:
1293b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, " DRM_UNIT\n");
1294b285192aSMauro Carvalho Chehab 			break;
1295b285192aSMauro Carvalho Chehab 		default:
1296b285192aSMauro Carvalho Chehab 			dprintk(DBGLVL_API, "default %d\n", hdr->subtype);
1297b285192aSMauro Carvalho Chehab 		}
1298b285192aSMauro Carvalho Chehab 
1299b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " 1.%x\n", hdr->len);
1300b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " 2.%x\n", hdr->type);
1301b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " 3.%x\n", hdr->subtype);
1302b285192aSMauro Carvalho Chehab 		dprintk(DBGLVL_API, " 4.%x\n", hdr->unitid);
1303b285192aSMauro Carvalho Chehab 
1304b285192aSMauro Carvalho Chehab 		idx += hdr->len;
1305b285192aSMauro Carvalho Chehab 	}
1306b285192aSMauro Carvalho Chehab 
1307b285192aSMauro Carvalho Chehab 	return 0;
1308b285192aSMauro Carvalho Chehab }
1309b285192aSMauro Carvalho Chehab 
saa7164_api_enum_subdevs(struct saa7164_dev * dev)1310b285192aSMauro Carvalho Chehab int saa7164_api_enum_subdevs(struct saa7164_dev *dev)
1311b285192aSMauro Carvalho Chehab {
1312b285192aSMauro Carvalho Chehab 	int ret;
1313b285192aSMauro Carvalho Chehab 	u32 buflen = 0;
1314b285192aSMauro Carvalho Chehab 	u8 *buf;
1315b285192aSMauro Carvalho Chehab 
1316b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s()\n", __func__);
1317b285192aSMauro Carvalho Chehab 
1318b285192aSMauro Carvalho Chehab 	/* Get the total descriptor length */
1319b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, 0, GET_LEN,
1320b285192aSMauro Carvalho Chehab 		GET_DESCRIPTORS_CONTROL, sizeof(buflen), &buflen);
1321b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
1322b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
1323b285192aSMauro Carvalho Chehab 
1324b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s() total descriptor size = %d bytes.\n",
1325b285192aSMauro Carvalho Chehab 		__func__, buflen);
1326b285192aSMauro Carvalho Chehab 
1327b285192aSMauro Carvalho Chehab 	/* Allocate enough storage for all of the descs */
1328b285192aSMauro Carvalho Chehab 	buf = kzalloc(buflen, GFP_KERNEL);
1329b285192aSMauro Carvalho Chehab 	if (!buf)
1330b285192aSMauro Carvalho Chehab 		return SAA_ERR_NO_RESOURCES;
1331b285192aSMauro Carvalho Chehab 
1332b285192aSMauro Carvalho Chehab 	/* Retrieve them */
1333b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, 0, GET_CUR,
1334b285192aSMauro Carvalho Chehab 		GET_DESCRIPTORS_CONTROL, buflen, buf);
1335b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK) {
1336b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n", __func__, ret);
1337b285192aSMauro Carvalho Chehab 		goto out;
1338b285192aSMauro Carvalho Chehab 	}
1339b285192aSMauro Carvalho Chehab 
1340b285192aSMauro Carvalho Chehab 	if (saa_debug & DBGLVL_API)
1341b285192aSMauro Carvalho Chehab 		print_hex_dump(KERN_INFO, "", DUMP_PREFIX_OFFSET, 16, 1, buf,
1342b285192aSMauro Carvalho Chehab 			       buflen & ~15, false);
1343b285192aSMauro Carvalho Chehab 
1344b285192aSMauro Carvalho Chehab 	saa7164_api_dump_subdevs(dev, buf, buflen);
1345b285192aSMauro Carvalho Chehab 
1346b285192aSMauro Carvalho Chehab out:
1347b285192aSMauro Carvalho Chehab 	kfree(buf);
1348b285192aSMauro Carvalho Chehab 	return ret;
1349b285192aSMauro Carvalho Chehab }
1350b285192aSMauro Carvalho Chehab 
saa7164_api_i2c_read(struct saa7164_i2c * bus,u8 addr,u32 reglen,u8 * reg,u32 datalen,u8 * data)1351b285192aSMauro Carvalho Chehab int saa7164_api_i2c_read(struct saa7164_i2c *bus, u8 addr, u32 reglen, u8 *reg,
1352b285192aSMauro Carvalho Chehab 	u32 datalen, u8 *data)
1353b285192aSMauro Carvalho Chehab {
1354b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = bus->dev;
1355b285192aSMauro Carvalho Chehab 	u16 len = 0;
1356b285192aSMauro Carvalho Chehab 	int unitid;
1357b285192aSMauro Carvalho Chehab 	u8 buf[256];
1358b285192aSMauro Carvalho Chehab 	int ret;
1359b285192aSMauro Carvalho Chehab 
13608ed5e4e8SSteven Toth 	dprintk(DBGLVL_API, "%s() addr=%x reglen=%d datalen=%d\n",
13618ed5e4e8SSteven Toth 		__func__, addr, reglen, datalen);
1362b285192aSMauro Carvalho Chehab 
1363b285192aSMauro Carvalho Chehab 	if (reglen > 4)
1364b285192aSMauro Carvalho Chehab 		return -EIO;
1365b285192aSMauro Carvalho Chehab 
1366b285192aSMauro Carvalho Chehab 	/* Prepare the send buffer */
1367b285192aSMauro Carvalho Chehab 	/* Bytes 00-03 source register length
1368b285192aSMauro Carvalho Chehab 	 *       04-07 source bytes to read
1369b285192aSMauro Carvalho Chehab 	 *       08... register address
1370b285192aSMauro Carvalho Chehab 	 */
1371b285192aSMauro Carvalho Chehab 	memset(buf, 0, sizeof(buf));
1372b285192aSMauro Carvalho Chehab 	memcpy((buf + 2 * sizeof(u32) + 0), reg, reglen);
1373b285192aSMauro Carvalho Chehab 	*((u32 *)(buf + 0 * sizeof(u32))) = reglen;
1374b285192aSMauro Carvalho Chehab 	*((u32 *)(buf + 1 * sizeof(u32))) = datalen;
1375b285192aSMauro Carvalho Chehab 
1376b285192aSMauro Carvalho Chehab 	unitid = saa7164_i2caddr_to_unitid(bus, addr);
1377b285192aSMauro Carvalho Chehab 	if (unitid < 0) {
1378b285192aSMauro Carvalho Chehab 		printk(KERN_ERR
1379b285192aSMauro Carvalho Chehab 			"%s() error, cannot translate regaddr 0x%x to unitid\n",
1380b285192aSMauro Carvalho Chehab 			__func__, addr);
1381b285192aSMauro Carvalho Chehab 		return -EIO;
1382b285192aSMauro Carvalho Chehab 	}
1383b285192aSMauro Carvalho Chehab 
1384b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(bus->dev, unitid, GET_LEN,
1385b285192aSMauro Carvalho Chehab 		EXU_REGISTER_ACCESS_CONTROL, sizeof(len), &len);
1386b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK) {
1387b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret(1) = 0x%x\n", __func__, ret);
1388b285192aSMauro Carvalho Chehab 		return -EIO;
1389b285192aSMauro Carvalho Chehab 	}
1390b285192aSMauro Carvalho Chehab 
1391b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s() len = %d bytes\n", __func__, len);
1392b285192aSMauro Carvalho Chehab 
1393b285192aSMauro Carvalho Chehab 	if (saa_debug & DBGLVL_I2C)
1394b285192aSMauro Carvalho Chehab 		print_hex_dump(KERN_INFO, "", DUMP_PREFIX_OFFSET, 16, 1, buf,
1395b285192aSMauro Carvalho Chehab 			       32, false);
1396b285192aSMauro Carvalho Chehab 
1397b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(bus->dev, unitid, GET_CUR,
1398b285192aSMauro Carvalho Chehab 		EXU_REGISTER_ACCESS_CONTROL, len, &buf);
1399b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
1400b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret(2) = 0x%x\n", __func__, ret);
1401b285192aSMauro Carvalho Chehab 	else {
1402b285192aSMauro Carvalho Chehab 		if (saa_debug & DBGLVL_I2C)
1403b285192aSMauro Carvalho Chehab 			print_hex_dump(KERN_INFO, "", DUMP_PREFIX_OFFSET, 16, 1,
1404b285192aSMauro Carvalho Chehab 				       buf, sizeof(buf), false);
1405b285192aSMauro Carvalho Chehab 		memcpy(data, (buf + 2 * sizeof(u32) + reglen), datalen);
1406b285192aSMauro Carvalho Chehab 	}
1407b285192aSMauro Carvalho Chehab 
1408b285192aSMauro Carvalho Chehab 	return ret == SAA_OK ? 0 : -EIO;
1409b285192aSMauro Carvalho Chehab }
1410b285192aSMauro Carvalho Chehab 
1411b285192aSMauro Carvalho Chehab /* For a given 8 bit i2c address device, write the buffer */
saa7164_api_i2c_write(struct saa7164_i2c * bus,u8 addr,u32 datalen,u8 * data)1412b285192aSMauro Carvalho Chehab int saa7164_api_i2c_write(struct saa7164_i2c *bus, u8 addr, u32 datalen,
1413b285192aSMauro Carvalho Chehab 	u8 *data)
1414b285192aSMauro Carvalho Chehab {
1415b285192aSMauro Carvalho Chehab 	struct saa7164_dev *dev = bus->dev;
1416b285192aSMauro Carvalho Chehab 	u16 len = 0;
1417b285192aSMauro Carvalho Chehab 	int unitid;
1418b285192aSMauro Carvalho Chehab 	int reglen;
1419b285192aSMauro Carvalho Chehab 	u8 buf[256];
1420b285192aSMauro Carvalho Chehab 	int ret;
1421b285192aSMauro Carvalho Chehab 
14228ed5e4e8SSteven Toth 	dprintk(DBGLVL_API, "%s() addr=0x%2x len=0x%x\n",
14238ed5e4e8SSteven Toth 		__func__, addr, datalen);
1424b285192aSMauro Carvalho Chehab 
1425b285192aSMauro Carvalho Chehab 	if ((datalen == 0) || (datalen > 232))
1426b285192aSMauro Carvalho Chehab 		return -EIO;
1427b285192aSMauro Carvalho Chehab 
1428b285192aSMauro Carvalho Chehab 	memset(buf, 0, sizeof(buf));
1429b285192aSMauro Carvalho Chehab 
1430b285192aSMauro Carvalho Chehab 	unitid = saa7164_i2caddr_to_unitid(bus, addr);
1431b285192aSMauro Carvalho Chehab 	if (unitid < 0) {
1432b285192aSMauro Carvalho Chehab 		printk(KERN_ERR
1433b285192aSMauro Carvalho Chehab 			"%s() error, cannot translate regaddr 0x%x to unitid\n",
1434b285192aSMauro Carvalho Chehab 			__func__, addr);
1435b285192aSMauro Carvalho Chehab 		return -EIO;
1436b285192aSMauro Carvalho Chehab 	}
1437b285192aSMauro Carvalho Chehab 
1438b285192aSMauro Carvalho Chehab 	reglen = saa7164_i2caddr_to_reglen(bus, addr);
1439b285192aSMauro Carvalho Chehab 	if (reglen < 0) {
1440b285192aSMauro Carvalho Chehab 		printk(KERN_ERR
1441b285192aSMauro Carvalho Chehab 			"%s() error, cannot translate regaddr to reglen\n",
1442b285192aSMauro Carvalho Chehab 			__func__);
1443b285192aSMauro Carvalho Chehab 		return -EIO;
1444b285192aSMauro Carvalho Chehab 	}
1445b285192aSMauro Carvalho Chehab 
1446b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(bus->dev, unitid, GET_LEN,
1447b285192aSMauro Carvalho Chehab 		EXU_REGISTER_ACCESS_CONTROL, sizeof(len), &len);
1448b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK) {
1449b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret(1) = 0x%x\n", __func__, ret);
1450b285192aSMauro Carvalho Chehab 		return -EIO;
1451b285192aSMauro Carvalho Chehab 	}
1452b285192aSMauro Carvalho Chehab 
14533600433fSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s() len = %d bytes unitid=0x%x\n", __func__,
14543600433fSMauro Carvalho Chehab 		len, unitid);
1455b285192aSMauro Carvalho Chehab 
1456b285192aSMauro Carvalho Chehab 	/* Prepare the send buffer */
1457b285192aSMauro Carvalho Chehab 	/* Bytes 00-03 dest register length
1458b285192aSMauro Carvalho Chehab 	 *       04-07 dest bytes to write
1459b285192aSMauro Carvalho Chehab 	 *       08... register address
1460b285192aSMauro Carvalho Chehab 	 */
1461b285192aSMauro Carvalho Chehab 	*((u32 *)(buf + 0 * sizeof(u32))) = reglen;
1462b285192aSMauro Carvalho Chehab 	*((u32 *)(buf + 1 * sizeof(u32))) = datalen - reglen;
1463b285192aSMauro Carvalho Chehab 	memcpy((buf + 2 * sizeof(u32)), data, datalen);
1464b285192aSMauro Carvalho Chehab 
1465b285192aSMauro Carvalho Chehab 	if (saa_debug & DBGLVL_I2C)
1466b285192aSMauro Carvalho Chehab 		print_hex_dump(KERN_INFO, "", DUMP_PREFIX_OFFSET, 16, 1,
1467b285192aSMauro Carvalho Chehab 			       buf, sizeof(buf), false);
1468b285192aSMauro Carvalho Chehab 
1469b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(bus->dev, unitid, SET_CUR,
1470b285192aSMauro Carvalho Chehab 		EXU_REGISTER_ACCESS_CONTROL, len, &buf);
1471b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
1472b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret(2) = 0x%x\n", __func__, ret);
1473b285192aSMauro Carvalho Chehab 
1474b285192aSMauro Carvalho Chehab 	return ret == SAA_OK ? 0 : -EIO;
1475b285192aSMauro Carvalho Chehab }
1476b285192aSMauro Carvalho Chehab 
saa7164_api_modify_gpio(struct saa7164_dev * dev,u8 unitid,u8 pin,u8 state)14775faf7db8SMauro Carvalho Chehab static int saa7164_api_modify_gpio(struct saa7164_dev *dev, u8 unitid,
1478b285192aSMauro Carvalho Chehab 	u8 pin, u8 state)
1479b285192aSMauro Carvalho Chehab {
1480b285192aSMauro Carvalho Chehab 	int ret;
1481b285192aSMauro Carvalho Chehab 	struct tmComResGPIO t;
1482b285192aSMauro Carvalho Chehab 
1483b285192aSMauro Carvalho Chehab 	dprintk(DBGLVL_API, "%s(0x%x, %d, %d)\n",
1484b285192aSMauro Carvalho Chehab 		__func__, unitid, pin, state);
1485b285192aSMauro Carvalho Chehab 
1486b285192aSMauro Carvalho Chehab 	if ((pin > 7) || (state > 2))
1487b285192aSMauro Carvalho Chehab 		return SAA_ERR_BAD_PARAMETER;
1488b285192aSMauro Carvalho Chehab 
1489b285192aSMauro Carvalho Chehab 	t.pin = pin;
1490b285192aSMauro Carvalho Chehab 	t.state = state;
1491b285192aSMauro Carvalho Chehab 
1492b285192aSMauro Carvalho Chehab 	ret = saa7164_cmd_send(dev, unitid, SET_CUR,
1493b285192aSMauro Carvalho Chehab 		EXU_GPIO_CONTROL, sizeof(t), &t);
1494b285192aSMauro Carvalho Chehab 	if (ret != SAA_OK)
1495b285192aSMauro Carvalho Chehab 		printk(KERN_ERR "%s() error, ret = 0x%x\n",
1496b285192aSMauro Carvalho Chehab 			__func__, ret);
1497b285192aSMauro Carvalho Chehab 
1498b285192aSMauro Carvalho Chehab 	return ret;
1499b285192aSMauro Carvalho Chehab }
1500b285192aSMauro Carvalho Chehab 
saa7164_api_set_gpiobit(struct saa7164_dev * dev,u8 unitid,u8 pin)1501b285192aSMauro Carvalho Chehab int saa7164_api_set_gpiobit(struct saa7164_dev *dev, u8 unitid,
1502b285192aSMauro Carvalho Chehab 	u8 pin)
1503b285192aSMauro Carvalho Chehab {
1504b285192aSMauro Carvalho Chehab 	return saa7164_api_modify_gpio(dev, unitid, pin, 1);
1505b285192aSMauro Carvalho Chehab }
1506b285192aSMauro Carvalho Chehab 
saa7164_api_clear_gpiobit(struct saa7164_dev * dev,u8 unitid,u8 pin)1507b285192aSMauro Carvalho Chehab int saa7164_api_clear_gpiobit(struct saa7164_dev *dev, u8 unitid,
1508b285192aSMauro Carvalho Chehab 	u8 pin)
1509b285192aSMauro Carvalho Chehab {
1510b285192aSMauro Carvalho Chehab 	return saa7164_api_modify_gpio(dev, unitid, pin, 0);
1511b285192aSMauro Carvalho Chehab }
1512b285192aSMauro Carvalho Chehab 
1513