On Mon, 24 Jul 2017 22:49:18 +0200, Robert Jarzmik wrote:
AC97 is a bus for sound usage. It enables for a AC97 AC-Link to link one controller to 0 to 4 AC97 codecs.
The goal of this new implementation is to implement a device/driver model for AC97, with an automatic scan of the bus and automatic discovery of AC97 codec devices.
Signed-off-by: Robert Jarzmik robert.jarzmik@free.fr
Since RFCv1:
Takashi's review
- changed the codec.h guard ... a better name could be found ...
 - added the AC97_* macros missing parenthesis
 - constantified the id_table in the codec driver structure
 - changed the 4 codecs linked list into an array
 - enabled the ac97 bus to be a module
 - added a slots_available to snd_ac97_controller_register() to have a way to prevent scanning and probing of unconnected codecs
 - removed useless ac97 bus index
 - all exported functions begin with snd_ac97_*()
 - change bus operations to controller+slot parameters instead of codec device
 Mark's review
- changed ac97_digital_controller into ac97_controller
 - rename ac97_digital_controller_*() into ac97_controller_*()
 - add the ac97 ac-link clock to the codec device (ie. the AC'97 BIT_CLK)
 Since RFCv2:
- more snd_ac97 namespace review
 - change the compat allocation prototype to force the user to provide and ac97_codec_device structure pointer
 Since v1:
- took into account all Lars comments
 Since v3:
- took into account Takashi's comments (ac97bus naming and module_exit)
 
It looks mostly OK, but some nitpicking:
diff --git a/include/sound/ac97/compat.h b/include/sound/ac97/compat.h new file mode 100644 index 000000000000..d876464bf7e4 --- /dev/null +++ b/include/sound/ac97/compat.h @@ -0,0 +1,21 @@ +/*
- Copyright (C) 2016 Robert Jarzmik robert.jarzmik@free.fr
 
- This program is free software; you can redistribute it and/or modify
 
- it under the terms of the GNU General Public License version 2 as
 
- published by the Free Software Foundation.
 
- This file is for backward compatibility with snd_ac97 structure and its
 
- multiple usages, such as the snd_ac97_bus and snd_ac97_build_ops.
 
- */
 +#ifndef AC97_COMPAT_H +#define AC97_COMPAT_H
+#include <sound/ac97_codec.h> +#include <sound/soc.h>
Is this inclusion needed? The code here doesn't look ASoC-specific at all.
--- /dev/null +++ b/include/sound/ac97/controller.h @@ -0,0 +1,84 @@ +/*
- Copyright (C) 2016 Robert Jarzmik robert.jarzmik@free.fr
 
- This program is free software; you can redistribute it and/or modify
 
- it under the terms of the GNU General Public License version 2 as
 
- published by the Free Software Foundation.
 - */
 +#ifndef AC97_CONTROLLER_H +#define AC97_CONTROLLER_H
+#include <linux/list.h>
+#define AC97_BUS_MAX_CODECS 4 +#define AC97_SLOTS_AVAILABLE_ALL 0xf
+struct device;
You need the definition struct device in below (it's no pointer), thus you have to include <linux/device.h> instead.
+/**
- struct ac97_controller - The AC97 controller of the AC-Link
 
- @ops: the AC97 operations.
 
- @controllers: linked list of all existing controllers.
 
- @adap: the shell device ac97-%d, ie. ac97 adapter
 
- @nr: the number of the shell device
 
- @parent: the device providing the AC97 controller.
 
- @slots_available: the mask of accessible/scanable codecs.
 
- @codecs: the 4 possible AC97 codecs (NULL if none found).
 
- @codecs_pdata: platform_data for each codec (NULL if no pdata).
 
- This structure is internal to AC97 bus, and should not be used by the
 
- controllers themselves, excepting for using @dev.
 - */
 +struct ac97_controller {
- const struct ac97_controller_ops *ops;
 
The struct isn't declared beforehand? GCC will warn.
- struct list_head controllers;
 - struct device adap;
 - int nr;
 - struct device *parent;
 - unsigned short slots_available;
 
I'd move parent field below, so that 64bit pointer can be aligned better.
+static int ac97_codec_add(struct ac97_controller *ac97_ctrl, int idx,
unsigned int vendor_id)+{
- struct ac97_codec_device *codec;
 - int ret;
 - codec = kzalloc(sizeof(*codec), GFP_KERNEL);
 - if (!codec)
 return -ENOMEM;- ac97_ctrl->codecs[idx] = codec;
 - codec->vendor_id = vendor_id;
 - codec->dev.release = ac97_codec_release;
 - codec->dev.bus = &ac97_bus_type;
 - codec->dev.parent = &ac97_ctrl->adap;
 - codec->num = idx;
 - codec->ac97_ctrl = ac97_ctrl;
 - device_initialize(&codec->dev);
 - dev_set_name(&codec->dev, "%s:%u", dev_name(ac97_ctrl->parent), idx);
 - ret = device_add(&codec->dev);
 - if (ret)
 goto err_free_codec;- return 0;
 +err_free_codec:
- kfree(codec);
 
This may leave the device name string. You need to call put_device() even if device_add() returns an error.
- ac97_ctrl->codecs[idx] = NULL;
 - return ret;
 +}
+unsigned int snd_ac97_bus_scan_one(struct ac97_controller *adrv,
unsigned int codec_num)+{
- unsigned short vid1, vid2;
 - int ret;
 - ret = adrv->ops->read(adrv, codec_num, AC97_VENDOR_ID1);
 - vid1 = (ret & 0xffff);
 - if (ret < 0)
 return 0;- ret = adrv->ops->read(adrv, codec_num, AC97_VENDOR_ID2);
 - vid2 = (ret & 0xffff);
 - if (ret < 0)
 return 0;- dev_dbg(&adrv->adap, "%s(codec_num=%u): vendor_id=0x%08x\n",
 __func__, codec_num, AC97_ID(vid1, vid2));- return AC97_ID(vid1, vid2);
 +}
+static int ac97_bus_scan(struct ac97_controller *ac97_ctrl) +{
- int ret, i;
 - unsigned int vendor_id;
 - for (i = 0; i < AC97_BUS_MAX_CODECS; i++) {
 if (ac97_codec_find(ac97_ctrl, i))continue;if (!(ac97_ctrl->slots_available & BIT(i)))continue;vendor_id = snd_ac97_bus_scan_one(ac97_ctrl, i);if (!vendor_id)continue;ret = ac97_codec_add(ac97_ctrl, i, vendor_id);if (ret < 0)return ret;- }
 - return 0;
 +}
+static int ac97_bus_reset(struct ac97_controller *ac97_ctrl) +{
- ac97_ctrl->ops->reset(ac97_ctrl);
 - return 0;
 +}
+/**
- snd_ac97_codec_driver_register - register an AC97 codec driver
 
- @dev: AC97 driver codec to register
 
- Register an AC97 codec driver to the ac97 bus driver, aka. the AC97 digital
 
- controller.
 
- Returns 0 on success or error code
 - */
 +int snd_ac97_codec_driver_register(struct ac97_codec_driver *drv) +{
- int ret;
 - drv->driver.bus = &ac97_bus_type;
 - ret = driver_register(&drv->driver);
 - return ret;
 
This can be simplified.
+} +EXPORT_SYMBOL(snd_ac97_codec_driver_register);
No GPL? (Ditto for other entries, too)
+int snd_ac97_reset(struct snd_ac97 *ac97, bool try_warm, unsigned int id,
- unsigned int id_mask)
 +{
- struct ac97_codec_device *adev = to_ac97_device(ac97->private_data);
 - struct ac97_controller *actrl = adev->ac97_ctrl;
 - if (try_warm) {
 compat_ac97_warm_reset(ac97);if (snd_ac97_bus_scan_one(actrl, adev->num) == adev->vendor_id)
Can we ignore id_mask here? I'm not quite sure whether it's fixed...
thanks,
Takashi