blob: 55791a0b3943c81f5dc5c65133bcc5a8d0d07827 [file] [log] [blame]
Liam Girdwood0ca06a02005-07-29 16:13:36 +02001/*
2 * Linux driver model AC97 bus interface
3 *
4 * Author: Nicolas Pitre
5 * Created: Jan 14, 2005
6 * Copyright: (C) MontaVista Software Inc.
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 */
13
Jaroslav Kysela5049c352005-08-22 12:19:14 +020014#include <linux/module.h>
Liam Girdwood0ca06a02005-07-29 16:13:36 +020015#include <linux/init.h>
16#include <linux/device.h>
17#include <linux/string.h>
Mark Brown96841ba2008-12-02 15:15:50 +000018#include <sound/ac97_codec.h>
Liam Girdwood0ca06a02005-07-29 16:13:36 +020019
20/*
Lars-Peter Clausen5f1d9802015-07-21 21:53:00 +020021 * snd_ac97_check_id() - Reads and checks the vendor ID of the device
22 * @ac97: The AC97 device to check
23 * @id: The ID to compare to
24 * @id_mask: Mask that is applied to the device ID before comparing to @id
25 *
26 * If @id is 0 this function returns true if the read device vendor ID is
27 * a valid ID. If @id is non 0 this functions returns true if @id
28 * matches the read vendor ID. Otherwise the function returns false.
29 */
30static bool snd_ac97_check_id(struct snd_ac97 *ac97, unsigned int id,
31 unsigned int id_mask)
32{
33 ac97->id = ac97->bus->ops->read(ac97, AC97_VENDOR_ID1) << 16;
34 ac97->id |= ac97->bus->ops->read(ac97, AC97_VENDOR_ID2);
35
36 if (ac97->id == 0x0 || ac97->id == 0xffffffff)
37 return false;
38
39 if (id != 0 && id != (ac97->id & id_mask))
40 return false;
41
42 return true;
43}
44
45/**
46 * snd_ac97_reset() - Reset AC'97 device
47 * @ac97: The AC'97 device to reset
48 * @try_warm: Try a warm reset first
49 * @id: Expected device vendor ID
50 * @id_mask: Mask that is applied to the device ID before comparing to @id
51 *
52 * This function resets the AC'97 device. If @try_warm is true the function
53 * first performs a warm reset. If the warm reset is successful the function
54 * returns 1. Otherwise or if @try_warm is false the function issues cold reset
55 * followed by a warm reset. If this is successful the function returns 0,
56 * otherwise a negative error code. If @id is 0 any valid device ID will be
57 * accepted, otherwise only the ID that matches @id and @id_mask is accepted.
58 */
59int snd_ac97_reset(struct snd_ac97 *ac97, bool try_warm, unsigned int id,
60 unsigned int id_mask)
61{
62 struct snd_ac97_bus_ops *ops = ac97->bus->ops;
63
64 if (try_warm && ops->warm_reset) {
65 ops->warm_reset(ac97);
66 if (snd_ac97_check_id(ac97, id, id_mask))
67 return 1;
68 }
69
70 if (ops->reset)
71 ops->reset(ac97);
72 if (ops->warm_reset)
73 ops->warm_reset(ac97);
74
75 if (snd_ac97_check_id(ac97, id, id_mask))
76 return 0;
77
78 return -ENODEV;
79}
80EXPORT_SYMBOL_GPL(snd_ac97_reset);
81
82/*
Nicolas Pitre3a91e952005-09-16 18:46:36 +020083 * Let drivers decide whether they want to support given codec from their
Jeffrin Josed0359c62010-12-06 19:27:53 +053084 * probe method. Drivers have direct access to the struct snd_ac97
85 * structure and may decide based on the id field amongst other things.
Liam Girdwood0ca06a02005-07-29 16:13:36 +020086 */
87static int ac97_bus_match(struct device *dev, struct device_driver *drv)
88{
Nicolas Pitre3a91e952005-09-16 18:46:36 +020089 return 1;
Liam Girdwood0ca06a02005-07-29 16:13:36 +020090}
91
Martin Langer66e27782007-02-05 13:02:35 +010092#ifdef CONFIG_PM
Liam Girdwood0ca06a02005-07-29 16:13:36 +020093static int ac97_bus_suspend(struct device *dev, pm_message_t state)
94{
95 int ret = 0;
96
Nicolas Pitre90b66e82005-09-16 18:50:53 +020097 if (dev->driver && dev->driver->suspend)
Russell King9480e302005-10-28 09:52:56 -070098 ret = dev->driver->suspend(dev, state);
99
Liam Girdwood0ca06a02005-07-29 16:13:36 +0200100 return ret;
101}
102
103static int ac97_bus_resume(struct device *dev)
104{
105 int ret = 0;
106
Nicolas Pitre90b66e82005-09-16 18:50:53 +0200107 if (dev->driver && dev->driver->resume)
Russell King9480e302005-10-28 09:52:56 -0700108 ret = dev->driver->resume(dev);
109
Liam Girdwood0ca06a02005-07-29 16:13:36 +0200110 return ret;
111}
Martin Langer66e27782007-02-05 13:02:35 +0100112#endif /* CONFIG_PM */
Liam Girdwood0ca06a02005-07-29 16:13:36 +0200113
114struct bus_type ac97_bus_type = {
115 .name = "ac97",
116 .match = ac97_bus_match,
Martin Langer66e27782007-02-05 13:02:35 +0100117#ifdef CONFIG_PM
Liam Girdwood0ca06a02005-07-29 16:13:36 +0200118 .suspend = ac97_bus_suspend,
119 .resume = ac97_bus_resume,
Martin Langer66e27782007-02-05 13:02:35 +0100120#endif /* CONFIG_PM */
Liam Girdwood0ca06a02005-07-29 16:13:36 +0200121};
122
123static int __init ac97_bus_init(void)
124{
125 return bus_register(&ac97_bus_type);
126}
127
128subsys_initcall(ac97_bus_init);
129
130static void __exit ac97_bus_exit(void)
131{
132 bus_unregister(&ac97_bus_type);
133}
134
135module_exit(ac97_bus_exit);
136
137EXPORT_SYMBOL(ac97_bus_type);
138
139MODULE_LICENSE("GPL");