|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/proc_fs.h>
|
|
|
|
#include <linux/ioport.h>
|
|
|
|
#include <linux/sysctl.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/i2c.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/soundcard.h>
|
|
|
|
#include <asm/uaccess.h>
|
|
|
|
#include <asm/errno.h>
|
|
|
|
#include <asm/io.h>
|
|
|
|
#include <asm/prom.h>
|
|
|
|
|
|
|
|
#include "tas_common.h"
|
|
|
|
|
|
|
|
#define CALL0(proc) \
|
|
|
|
do { \
|
|
|
|
struct tas_data_t *self; \
|
|
|
|
if (!tas_client || driver_hooks == NULL) \
|
|
|
|
return -1; \
|
|
|
|
self = dev_get_drvdata(&tas_client->dev); \
|
|
|
|
if (driver_hooks->proc) \
|
|
|
|
return driver_hooks->proc(self); \
|
|
|
|
else \
|
|
|
|
return -EINVAL; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define CALL(proc,arg...) \
|
|
|
|
do { \
|
|
|
|
struct tas_data_t *self; \
|
|
|
|
if (!tas_client || driver_hooks == NULL) \
|
|
|
|
return -1; \
|
|
|
|
self = dev_get_drvdata(&tas_client->dev); \
|
|
|
|
if (driver_hooks->proc) \
|
|
|
|
return driver_hooks->proc(self, ## arg); \
|
|
|
|
else \
|
|
|
|
return -EINVAL; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
static u8 tas_i2c_address = 0x34;
|
|
|
|
static struct i2c_client *tas_client;
|
|
|
|
static struct device_node* tas_node;
|
|
|
|
|
|
|
|
static int tas_attach_adapter(struct i2c_adapter *);
|
|
|
|
static int tas_detach_client(struct i2c_client *);
|
|
|
|
|
|
|
|
struct i2c_driver tas_driver = {
|
|
|
|
.driver = {
|
|
|
|
.name = "tas",
|
|
|
|
},
|
|
|
|
.attach_adapter = tas_attach_adapter,
|
|
|
|
.detach_client = tas_detach_client,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct tas_driver_hooks_t *driver_hooks;
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_register_driver(struct tas_driver_hooks_t *hooks)
|
|
|
|
{
|
|
|
|
driver_hooks = hooks;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_get_mixer_level(int mixer, uint *level)
|
|
|
|
{
|
|
|
|
CALL(get_mixer_level,mixer,level);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_set_mixer_level(int mixer,uint level)
|
|
|
|
{
|
|
|
|
CALL(set_mixer_level,mixer,level);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_enter_sleep(void)
|
|
|
|
{
|
|
|
|
CALL0(enter_sleep);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_leave_sleep(void)
|
|
|
|
{
|
|
|
|
CALL0(leave_sleep);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_supported_mixers(void)
|
|
|
|
{
|
|
|
|
CALL0(supported_mixers);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_mixer_is_stereo(int mixer)
|
|
|
|
{
|
|
|
|
CALL(mixer_is_stereo,mixer);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_stereo_mixers(void)
|
|
|
|
{
|
|
|
|
CALL0(stereo_mixers);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_output_device_change(int device_id,int layout_id,int speaker_id)
|
|
|
|
{
|
|
|
|
CALL(output_device_change,device_id,layout_id,speaker_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_device_ioctl(u_int cmd, u_long arg)
|
|
|
|
{
|
|
|
|
CALL(device_ioctl,cmd,arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tas_post_init(void)
|
|
|
|
{
|
|
|
|
CALL0(post_init);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
tas_detect_client(struct i2c_adapter *adapter, int address)
|
|
|
|
{
|
|
|
|
static const char *client_name = "tas Digital Equalizer";
|
|
|
|
struct i2c_client *new_client;
|
|
|
|
int rc = -ENODEV;
|
|
|
|
|
|
|
|
if (!driver_hooks) {
|
|
|
|
printk(KERN_ERR "tas_detect_client called with no hooks !\n");
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_client = kmalloc(sizeof(*new_client), GFP_KERNEL);
|
|
|
|
if (!new_client)
|
|
|
|
return -ENOMEM;
|
|
|
|
memset(new_client, 0, sizeof(*new_client));
|
|
|
|
|
|
|
|
new_client->addr = address;
|
|
|
|
new_client->adapter = adapter;
|
|
|
|
new_client->driver = &tas_driver;
|
|
|
|
strlcpy(new_client->name, client_name, DEVICE_NAME_SIZE);
|
|
|
|
|
|
|
|
if (driver_hooks->init(new_client))
|
|
|
|
goto bail;
|
|
|
|
|
|
|
|
/* Tell the i2c layer a new client has arrived */
|
|
|
|
if (i2c_attach_client(new_client)) {
|
|
|
|
driver_hooks->uninit(dev_get_drvdata(&new_client->dev));
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
|
|
|
|
tas_client = new_client;
|
|
|
|
return 0;
|
|
|
|
bail:
|
|
|
|
tas_client = NULL;
|
|
|
|
kfree(new_client);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
tas_attach_adapter(struct i2c_adapter *adapter)
|
|
|
|
{
|
|
|
|
if (!strncmp(adapter->name, "mac-io", 6))
|
|
|
|
return tas_detect_client(adapter, tas_i2c_address);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
tas_detach_client(struct i2c_client *client)
|
|
|
|
{
|
|
|
|
if (client == tas_client) {
|
|
|
|
driver_hooks->uninit(dev_get_drvdata(&client->dev));
|
|
|
|
|
|
|
|
i2c_detach_client(client);
|
|
|
|
kfree(client);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
tas_cleanup(void)
|
|
|
|
{
|
|
|
|
i2c_del_driver(&tas_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
int __init
|
|
|
|
tas_init(int driver_id, const char *driver_name)
|
|
|
|
{
|
|
|
|
u32* paddr;
|
|
|
|
|
|
|
|
printk(KERN_INFO "tas driver [%s])\n", driver_name);
|
|
|
|
|
|
|
|
#ifndef CONFIG_I2C_POWERMAC
|
|
|
|
request_module("i2c-powermac");
|
|
|
|
#endif
|
|
|
|
tas_node = find_devices("deq");
|
|
|
|
if (tas_node == NULL)
|
|
|
|
return -ENODEV;
|
|
|
|
paddr = (u32 *)get_property(tas_node, "i2c-address", NULL);
|
|
|
|
if (paddr) {
|
|
|
|
tas_i2c_address = (*paddr) >> 1;
|
|
|
|
printk(KERN_INFO "using i2c address: 0x%x from device-tree\n",
|
|
|
|
tas_i2c_address);
|
|
|
|
} else
|
|
|
|
printk(KERN_INFO "using i2c address: 0x%x (default)\n",
|
|
|
|
tas_i2c_address);
|
|
|
|
|
|
|
|
return i2c_add_driver(&tas_driver);
|
|
|
|
}
|