162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * 462306a36Sopenharmony_ci * Copyright (C) 2005 Mike Isely <isely@pobox.com> 562306a36Sopenharmony_ci * Copyright (C) 2004 Aurelien Alleaume <slts@free.fr> 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci/* 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci This source file is specifically designed to interface with the 1162306a36Sopenharmony_ci saa711x support that is available in the v4l available starting 1262306a36Sopenharmony_ci with linux 2.6.15. 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci*/ 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci#include "pvrusb2-video-v4l.h" 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci#include "pvrusb2-hdw-internal.h" 2162306a36Sopenharmony_ci#include "pvrusb2-debug.h" 2262306a36Sopenharmony_ci#include <linux/videodev2.h> 2362306a36Sopenharmony_ci#include <media/v4l2-common.h> 2462306a36Sopenharmony_ci#include <media/i2c/saa7115.h> 2562306a36Sopenharmony_ci#include <linux/errno.h> 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_cistruct routing_scheme { 2862306a36Sopenharmony_ci const int *def; 2962306a36Sopenharmony_ci unsigned int cnt; 3062306a36Sopenharmony_ci}; 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_cistatic const int routing_scheme0[] = { 3462306a36Sopenharmony_ci [PVR2_CVAL_INPUT_TV] = SAA7115_COMPOSITE4, 3562306a36Sopenharmony_ci /* In radio mode, we mute the video, but point at one 3662306a36Sopenharmony_ci spot just to stay consistent */ 3762306a36Sopenharmony_ci [PVR2_CVAL_INPUT_RADIO] = SAA7115_COMPOSITE5, 3862306a36Sopenharmony_ci [PVR2_CVAL_INPUT_COMPOSITE] = SAA7115_COMPOSITE5, 3962306a36Sopenharmony_ci [PVR2_CVAL_INPUT_SVIDEO] = SAA7115_SVIDEO2, 4062306a36Sopenharmony_ci}; 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_cistatic const struct routing_scheme routing_def0 = { 4362306a36Sopenharmony_ci .def = routing_scheme0, 4462306a36Sopenharmony_ci .cnt = ARRAY_SIZE(routing_scheme0), 4562306a36Sopenharmony_ci}; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_cistatic const int routing_scheme1[] = { 4862306a36Sopenharmony_ci [PVR2_CVAL_INPUT_TV] = SAA7115_COMPOSITE4, 4962306a36Sopenharmony_ci [PVR2_CVAL_INPUT_RADIO] = SAA7115_COMPOSITE5, 5062306a36Sopenharmony_ci [PVR2_CVAL_INPUT_COMPOSITE] = SAA7115_COMPOSITE3, 5162306a36Sopenharmony_ci [PVR2_CVAL_INPUT_SVIDEO] = SAA7115_SVIDEO2, /* or SVIDEO0, it seems */ 5262306a36Sopenharmony_ci}; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_cistatic const struct routing_scheme routing_def1 = { 5562306a36Sopenharmony_ci .def = routing_scheme1, 5662306a36Sopenharmony_ci .cnt = ARRAY_SIZE(routing_scheme1), 5762306a36Sopenharmony_ci}; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_cistatic const struct routing_scheme *routing_schemes[] = { 6062306a36Sopenharmony_ci [PVR2_ROUTING_SCHEME_HAUPPAUGE] = &routing_def0, 6162306a36Sopenharmony_ci [PVR2_ROUTING_SCHEME_ONAIR] = &routing_def1, 6262306a36Sopenharmony_ci}; 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_civoid pvr2_saa7115_subdev_update(struct pvr2_hdw *hdw, struct v4l2_subdev *sd) 6562306a36Sopenharmony_ci{ 6662306a36Sopenharmony_ci if (hdw->input_dirty || hdw->force_dirty) { 6762306a36Sopenharmony_ci const struct routing_scheme *sp; 6862306a36Sopenharmony_ci unsigned int sid = hdw->hdw_desc->signal_routing_scheme; 6962306a36Sopenharmony_ci u32 input; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci pvr2_trace(PVR2_TRACE_CHIPS, "subdev v4l2 set_input(%d)", 7262306a36Sopenharmony_ci hdw->input_val); 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_ci sp = (sid < ARRAY_SIZE(routing_schemes)) ? 7562306a36Sopenharmony_ci routing_schemes[sid] : NULL; 7662306a36Sopenharmony_ci if ((sp == NULL) || 7762306a36Sopenharmony_ci (hdw->input_val < 0) || 7862306a36Sopenharmony_ci (hdw->input_val >= sp->cnt)) { 7962306a36Sopenharmony_ci pvr2_trace(PVR2_TRACE_ERROR_LEGS, 8062306a36Sopenharmony_ci "*** WARNING *** subdev v4l2 set_input: Invalid routing scheme (%u) and/or input (%d)", 8162306a36Sopenharmony_ci sid, hdw->input_val); 8262306a36Sopenharmony_ci return; 8362306a36Sopenharmony_ci } 8462306a36Sopenharmony_ci input = sp->def[hdw->input_val]; 8562306a36Sopenharmony_ci sd->ops->video->s_routing(sd, input, 0, 0); 8662306a36Sopenharmony_ci } 8762306a36Sopenharmony_ci} 88