162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * tef6862.c Philips TEF6862 Car Radio Enhanced Selectivity Tuner 462306a36Sopenharmony_ci * Copyright (c) 2009 Intel Corporation 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/module.h> 862306a36Sopenharmony_ci#include <linux/init.h> 962306a36Sopenharmony_ci#include <linux/errno.h> 1062306a36Sopenharmony_ci#include <linux/kernel.h> 1162306a36Sopenharmony_ci#include <linux/interrupt.h> 1262306a36Sopenharmony_ci#include <linux/i2c.h> 1362306a36Sopenharmony_ci#include <linux/slab.h> 1462306a36Sopenharmony_ci#include <media/v4l2-ioctl.h> 1562306a36Sopenharmony_ci#include <media/v4l2-device.h> 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#define DRIVER_NAME "tef6862" 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci#define FREQ_MUL 16000 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#define TEF6862_LO_FREQ (875U * FREQ_MUL / 10) 2262306a36Sopenharmony_ci#define TEF6862_HI_FREQ (108U * FREQ_MUL) 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci/* Write mode sub addresses */ 2562306a36Sopenharmony_ci#define WM_SUB_BANDWIDTH 0x0 2662306a36Sopenharmony_ci#define WM_SUB_PLLM 0x1 2762306a36Sopenharmony_ci#define WM_SUB_PLLL 0x2 2862306a36Sopenharmony_ci#define WM_SUB_DAA 0x3 2962306a36Sopenharmony_ci#define WM_SUB_AGC 0x4 3062306a36Sopenharmony_ci#define WM_SUB_BAND 0x5 3162306a36Sopenharmony_ci#define WM_SUB_CONTROL 0x6 3262306a36Sopenharmony_ci#define WM_SUB_LEVEL 0x7 3362306a36Sopenharmony_ci#define WM_SUB_IFCF 0x8 3462306a36Sopenharmony_ci#define WM_SUB_IFCAP 0x9 3562306a36Sopenharmony_ci#define WM_SUB_ACD 0xA 3662306a36Sopenharmony_ci#define WM_SUB_TEST 0xF 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_ci/* Different modes of the MSA register */ 3962306a36Sopenharmony_ci#define MSA_MODE_BUFFER 0x0 4062306a36Sopenharmony_ci#define MSA_MODE_PRESET 0x1 4162306a36Sopenharmony_ci#define MSA_MODE_SEARCH 0x2 4262306a36Sopenharmony_ci#define MSA_MODE_AF_UPDATE 0x3 4362306a36Sopenharmony_ci#define MSA_MODE_JUMP 0x4 4462306a36Sopenharmony_ci#define MSA_MODE_CHECK 0x5 4562306a36Sopenharmony_ci#define MSA_MODE_LOAD 0x6 4662306a36Sopenharmony_ci#define MSA_MODE_END 0x7 4762306a36Sopenharmony_ci#define MSA_MODE_SHIFT 5 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_cistruct tef6862_state { 5062306a36Sopenharmony_ci struct v4l2_subdev sd; 5162306a36Sopenharmony_ci unsigned long freq; 5262306a36Sopenharmony_ci}; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_cistatic inline struct tef6862_state *to_state(struct v4l2_subdev *sd) 5562306a36Sopenharmony_ci{ 5662306a36Sopenharmony_ci return container_of(sd, struct tef6862_state, sd); 5762306a36Sopenharmony_ci} 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_cistatic u16 tef6862_sigstr(struct i2c_client *client) 6062306a36Sopenharmony_ci{ 6162306a36Sopenharmony_ci u8 buf[4]; 6262306a36Sopenharmony_ci int err = i2c_master_recv(client, buf, sizeof(buf)); 6362306a36Sopenharmony_ci if (err == sizeof(buf)) 6462306a36Sopenharmony_ci return buf[3] << 8; 6562306a36Sopenharmony_ci return 0; 6662306a36Sopenharmony_ci} 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_cistatic int tef6862_g_tuner(struct v4l2_subdev *sd, struct v4l2_tuner *v) 6962306a36Sopenharmony_ci{ 7062306a36Sopenharmony_ci if (v->index > 0) 7162306a36Sopenharmony_ci return -EINVAL; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci /* only support FM for now */ 7462306a36Sopenharmony_ci strscpy(v->name, "FM", sizeof(v->name)); 7562306a36Sopenharmony_ci v->type = V4L2_TUNER_RADIO; 7662306a36Sopenharmony_ci v->rangelow = TEF6862_LO_FREQ; 7762306a36Sopenharmony_ci v->rangehigh = TEF6862_HI_FREQ; 7862306a36Sopenharmony_ci v->rxsubchans = V4L2_TUNER_SUB_MONO; 7962306a36Sopenharmony_ci v->capability = V4L2_TUNER_CAP_LOW; 8062306a36Sopenharmony_ci v->audmode = V4L2_TUNER_MODE_STEREO; 8162306a36Sopenharmony_ci v->signal = tef6862_sigstr(v4l2_get_subdevdata(sd)); 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci return 0; 8462306a36Sopenharmony_ci} 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_cistatic int tef6862_s_tuner(struct v4l2_subdev *sd, const struct v4l2_tuner *v) 8762306a36Sopenharmony_ci{ 8862306a36Sopenharmony_ci return v->index ? -EINVAL : 0; 8962306a36Sopenharmony_ci} 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistatic int tef6862_s_frequency(struct v4l2_subdev *sd, const struct v4l2_frequency *f) 9262306a36Sopenharmony_ci{ 9362306a36Sopenharmony_ci struct tef6862_state *state = to_state(sd); 9462306a36Sopenharmony_ci struct i2c_client *client = v4l2_get_subdevdata(sd); 9562306a36Sopenharmony_ci unsigned freq = f->frequency; 9662306a36Sopenharmony_ci u16 pll; 9762306a36Sopenharmony_ci u8 i2cmsg[3]; 9862306a36Sopenharmony_ci int err; 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci if (f->tuner != 0) 10162306a36Sopenharmony_ci return -EINVAL; 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_ci freq = clamp(freq, TEF6862_LO_FREQ, TEF6862_HI_FREQ); 10462306a36Sopenharmony_ci pll = 1964 + ((freq - TEF6862_LO_FREQ) * 20) / FREQ_MUL; 10562306a36Sopenharmony_ci i2cmsg[0] = (MSA_MODE_PRESET << MSA_MODE_SHIFT) | WM_SUB_PLLM; 10662306a36Sopenharmony_ci i2cmsg[1] = (pll >> 8) & 0xff; 10762306a36Sopenharmony_ci i2cmsg[2] = pll & 0xff; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ci err = i2c_master_send(client, i2cmsg, sizeof(i2cmsg)); 11062306a36Sopenharmony_ci if (err != sizeof(i2cmsg)) 11162306a36Sopenharmony_ci return err < 0 ? err : -EIO; 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci state->freq = freq; 11462306a36Sopenharmony_ci return 0; 11562306a36Sopenharmony_ci} 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_cistatic int tef6862_g_frequency(struct v4l2_subdev *sd, struct v4l2_frequency *f) 11862306a36Sopenharmony_ci{ 11962306a36Sopenharmony_ci struct tef6862_state *state = to_state(sd); 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci if (f->tuner != 0) 12262306a36Sopenharmony_ci return -EINVAL; 12362306a36Sopenharmony_ci f->type = V4L2_TUNER_RADIO; 12462306a36Sopenharmony_ci f->frequency = state->freq; 12562306a36Sopenharmony_ci return 0; 12662306a36Sopenharmony_ci} 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_cistatic const struct v4l2_subdev_tuner_ops tef6862_tuner_ops = { 12962306a36Sopenharmony_ci .g_tuner = tef6862_g_tuner, 13062306a36Sopenharmony_ci .s_tuner = tef6862_s_tuner, 13162306a36Sopenharmony_ci .s_frequency = tef6862_s_frequency, 13262306a36Sopenharmony_ci .g_frequency = tef6862_g_frequency, 13362306a36Sopenharmony_ci}; 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_cistatic const struct v4l2_subdev_ops tef6862_ops = { 13662306a36Sopenharmony_ci .tuner = &tef6862_tuner_ops, 13762306a36Sopenharmony_ci}; 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci/* 14062306a36Sopenharmony_ci * Generic i2c probe 14162306a36Sopenharmony_ci * concerning the addresses: i2c wants 7 bit (without the r/w bit), so '>>1' 14262306a36Sopenharmony_ci */ 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_cistatic int tef6862_probe(struct i2c_client *client) 14562306a36Sopenharmony_ci{ 14662306a36Sopenharmony_ci struct tef6862_state *state; 14762306a36Sopenharmony_ci struct v4l2_subdev *sd; 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci /* Check if the adapter supports the needed features */ 15062306a36Sopenharmony_ci if (!i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_BYTE_DATA)) 15162306a36Sopenharmony_ci return -EIO; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci v4l_info(client, "chip found @ 0x%02x (%s)\n", 15462306a36Sopenharmony_ci client->addr << 1, client->adapter->name); 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_ci state = kzalloc(sizeof(struct tef6862_state), GFP_KERNEL); 15762306a36Sopenharmony_ci if (state == NULL) 15862306a36Sopenharmony_ci return -ENOMEM; 15962306a36Sopenharmony_ci state->freq = TEF6862_LO_FREQ; 16062306a36Sopenharmony_ci 16162306a36Sopenharmony_ci sd = &state->sd; 16262306a36Sopenharmony_ci v4l2_i2c_subdev_init(sd, client, &tef6862_ops); 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci return 0; 16562306a36Sopenharmony_ci} 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_cistatic void tef6862_remove(struct i2c_client *client) 16862306a36Sopenharmony_ci{ 16962306a36Sopenharmony_ci struct v4l2_subdev *sd = i2c_get_clientdata(client); 17062306a36Sopenharmony_ci 17162306a36Sopenharmony_ci v4l2_device_unregister_subdev(sd); 17262306a36Sopenharmony_ci kfree(to_state(sd)); 17362306a36Sopenharmony_ci} 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_cistatic const struct i2c_device_id tef6862_id[] = { 17662306a36Sopenharmony_ci {DRIVER_NAME, 0}, 17762306a36Sopenharmony_ci {}, 17862306a36Sopenharmony_ci}; 17962306a36Sopenharmony_ci 18062306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, tef6862_id); 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_cistatic struct i2c_driver tef6862_driver = { 18362306a36Sopenharmony_ci .driver = { 18462306a36Sopenharmony_ci .name = DRIVER_NAME, 18562306a36Sopenharmony_ci }, 18662306a36Sopenharmony_ci .probe = tef6862_probe, 18762306a36Sopenharmony_ci .remove = tef6862_remove, 18862306a36Sopenharmony_ci .id_table = tef6862_id, 18962306a36Sopenharmony_ci}; 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_cimodule_i2c_driver(tef6862_driver); 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_ciMODULE_DESCRIPTION("TEF6862 Car Radio Enhanced Selectivity Tuner"); 19462306a36Sopenharmony_ciMODULE_AUTHOR("Mocean Laboratories"); 19562306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 196