blob: 61e51e35ba168547d000e99b19769ee170a4d5fa [file] [log] [blame]
Thomas Gleixner5765e782019-05-31 01:09:25 -07001// SPDX-License-Identifier: GPL-2.0-only
Wai Yew CHAY8cc72362009-05-14 08:05:58 +02002/**
3 * Copyright (C) 2008, Creative Technology Ltd. All Rights Reserved.
4 *
Wai Yew CHAY8cc72362009-05-14 08:05:58 +02005 * @File ctresource.c
6 *
7 * @Brief
8 * This file contains the implementation of some generic helper functions.
9 *
10 * @Author Liu Chun
11 * @Date May 15 2008
Wai Yew CHAY8cc72362009-05-14 08:05:58 +020012 */
13
14#include "ctresource.h"
15#include "cthardware.h"
16#include <linux/err.h>
17#include <linux/slab.h>
18
19#define AUDIO_SLOT_BLOCK_NUM 256
20
21/* Resource allocation based on bit-map management mechanism */
22static int
23get_resource(u8 *rscs, unsigned int amount,
24 unsigned int multi, unsigned int *ridx)
25{
Takashi Iwai514eef92009-06-08 14:57:57 +020026 int i, j, k, n;
Wai Yew CHAY8cc72362009-05-14 08:05:58 +020027
28 /* Check whether there are sufficient resources to meet request. */
29 for (i = 0, n = multi; i < amount; i++) {
30 j = i / 8;
31 k = i % 8;
32 if (rscs[j] & ((u8)1 << k)) {
33 n = multi;
34 continue;
35 }
36 if (!(--n))
37 break; /* found sufficient contiguous resources */
38 }
39
40 if (i >= amount) {
41 /* Can not find sufficient contiguous resources */
42 return -ENOENT;
43 }
44
45 /* Mark the contiguous bits in resource bit-map as used */
46 for (n = multi; n > 0; n--) {
47 j = i / 8;
48 k = i % 8;
49 rscs[j] |= ((u8)1 << k);
50 i--;
51 }
52
53 *ridx = i + 1;
54
55 return 0;
56}
57
58static int put_resource(u8 *rscs, unsigned int multi, unsigned int idx)
59{
Takashi Iwai514eef92009-06-08 14:57:57 +020060 unsigned int i, j, k, n;
Wai Yew CHAY8cc72362009-05-14 08:05:58 +020061
62 /* Mark the contiguous bits in resource bit-map as used */
63 for (n = multi, i = idx; n > 0; n--) {
64 j = i / 8;
65 k = i % 8;
66 rscs[j] &= ~((u8)1 << k);
67 i++;
68 }
69
70 return 0;
71}
72
73int mgr_get_resource(struct rsc_mgr *mgr, unsigned int n, unsigned int *ridx)
74{
Takashi Iwai514eef92009-06-08 14:57:57 +020075 int err;
Wai Yew CHAY8cc72362009-05-14 08:05:58 +020076
77 if (n > mgr->avail)
78 return -ENOENT;
79
80 err = get_resource(mgr->rscs, mgr->amount, n, ridx);
81 if (!err)
82 mgr->avail -= n;
83
84 return err;
85}
86
87int mgr_put_resource(struct rsc_mgr *mgr, unsigned int n, unsigned int idx)
88{
89 put_resource(mgr->rscs, n, idx);
90 mgr->avail += n;
91
92 return 0;
93}
94
Takashi Iwai6e0e75d2020-01-05 15:47:46 +010095static const unsigned char offset_in_audio_slot_block[NUM_RSCTYP] = {
Wai Yew CHAY8cc72362009-05-14 08:05:58 +020096 /* SRC channel is at Audio Ring slot 1 every 16 slots. */
97 [SRC] = 0x1,
98 [AMIXER] = 0x4,
99 [SUM] = 0xc,
100};
101
102static int rsc_index(const struct rsc *rsc)
103{
104 return rsc->conj;
105}
106
107static int audio_ring_slot(const struct rsc *rsc)
108{
109 return (rsc->conj << 4) + offset_in_audio_slot_block[rsc->type];
110}
111
112static int rsc_next_conj(struct rsc *rsc)
113{
114 unsigned int i;
115 for (i = 0; (i < 8) && (!(rsc->msr & (0x1 << i))); )
116 i++;
117 rsc->conj += (AUDIO_SLOT_BLOCK_NUM >> i);
118 return rsc->conj;
119}
120
121static int rsc_master(struct rsc *rsc)
122{
123 return rsc->conj = rsc->idx;
124}
125
Julia Lawall43f2cde2015-11-08 23:40:41 +0100126static const struct rsc_ops rsc_generic_ops = {
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200127 .index = rsc_index,
128 .output_slot = audio_ring_slot,
129 .master = rsc_master,
130 .next_conj = rsc_next_conj,
131};
132
Sudip Mukherjee66640892014-09-29 14:33:21 +0530133int
134rsc_init(struct rsc *rsc, u32 idx, enum RSCTYP type, u32 msr, struct hw *hw)
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200135{
136 int err = 0;
137
138 rsc->idx = idx;
139 rsc->conj = idx;
140 rsc->type = type;
141 rsc->msr = msr;
142 rsc->hw = hw;
143 rsc->ops = &rsc_generic_ops;
Takashi Iwai35ebf6e2009-07-22 17:12:34 +0200144 if (!hw) {
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200145 rsc->ctrl_blk = NULL;
146 return 0;
147 }
148
149 switch (type) {
150 case SRC:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530151 err = hw->src_rsc_get_ctrl_blk(&rsc->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200152 break;
153 case AMIXER:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530154 err = hw->amixer_rsc_get_ctrl_blk(&rsc->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200155 break;
156 case SRCIMP:
157 case SUM:
158 case DAIO:
159 break;
160 default:
Sudip Mukherjee0cae90a2014-09-29 14:33:26 +0530161 dev_err(((struct hw *)hw)->card->dev,
162 "Invalid resource type value %d!\n", type);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200163 return -EINVAL;
164 }
165
166 if (err) {
Sudip Mukherjee0cae90a2014-09-29 14:33:26 +0530167 dev_err(((struct hw *)hw)->card->dev,
168 "Failed to get resource control block!\n");
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200169 return err;
170 }
171
172 return 0;
173}
174
175int rsc_uninit(struct rsc *rsc)
176{
177 if ((NULL != rsc->hw) && (NULL != rsc->ctrl_blk)) {
178 switch (rsc->type) {
179 case SRC:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530180 rsc->hw->src_rsc_put_ctrl_blk(rsc->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200181 break;
182 case AMIXER:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530183 rsc->hw->amixer_rsc_put_ctrl_blk(rsc->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200184 break;
185 case SUM:
186 case DAIO:
187 break;
188 default:
Sudip Mukherjee0cae90a2014-09-29 14:33:26 +0530189 dev_err(((struct hw *)rsc->hw)->card->dev,
190 "Invalid resource type value %d!\n",
Sudip Mukherjee62afa852014-08-26 19:01:43 +0530191 rsc->type);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200192 break;
193 }
194
195 rsc->hw = rsc->ctrl_blk = NULL;
196 }
197
198 rsc->idx = rsc->conj = 0;
199 rsc->type = NUM_RSCTYP;
200 rsc->msr = 0;
201
202 return 0;
203}
204
205int rsc_mgr_init(struct rsc_mgr *mgr, enum RSCTYP type,
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530206 unsigned int amount, struct hw *hw)
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200207{
208 int err = 0;
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200209
210 mgr->type = NUM_RSCTYP;
211
212 mgr->rscs = kzalloc(((amount + 8 - 1) / 8), GFP_KERNEL);
Takashi Iwai35ebf6e2009-07-22 17:12:34 +0200213 if (!mgr->rscs)
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200214 return -ENOMEM;
215
216 switch (type) {
217 case SRC:
218 err = hw->src_mgr_get_ctrl_blk(&mgr->ctrl_blk);
219 break;
220 case SRCIMP:
221 err = hw->srcimp_mgr_get_ctrl_blk(&mgr->ctrl_blk);
222 break;
223 case AMIXER:
224 err = hw->amixer_mgr_get_ctrl_blk(&mgr->ctrl_blk);
225 break;
226 case DAIO:
227 err = hw->daio_mgr_get_ctrl_blk(hw, &mgr->ctrl_blk);
228 break;
229 case SUM:
230 break;
231 default:
Sudip Mukherjee0cae90a2014-09-29 14:33:26 +0530232 dev_err(hw->card->dev,
233 "Invalid resource type value %d!\n", type);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200234 err = -EINVAL;
235 goto error;
236 }
237
238 if (err) {
Sudip Mukherjee0cae90a2014-09-29 14:33:26 +0530239 dev_err(hw->card->dev,
240 "Failed to get manager control block!\n");
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200241 goto error;
242 }
243
244 mgr->type = type;
245 mgr->avail = mgr->amount = amount;
246 mgr->hw = hw;
247
248 return 0;
249
250error:
251 kfree(mgr->rscs);
252 return err;
253}
254
255int rsc_mgr_uninit(struct rsc_mgr *mgr)
256{
Himanshu Jha39cdc622017-08-31 20:36:42 +0530257 kfree(mgr->rscs);
258 mgr->rscs = NULL;
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200259
260 if ((NULL != mgr->hw) && (NULL != mgr->ctrl_blk)) {
261 switch (mgr->type) {
262 case SRC:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530263 mgr->hw->src_mgr_put_ctrl_blk(mgr->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200264 break;
265 case SRCIMP:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530266 mgr->hw->srcimp_mgr_put_ctrl_blk(mgr->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200267 break;
268 case AMIXER:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530269 mgr->hw->amixer_mgr_put_ctrl_blk(mgr->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200270 break;
271 case DAIO:
Sudip Mukherjeeb6bfe862014-09-29 14:33:22 +0530272 mgr->hw->daio_mgr_put_ctrl_blk(mgr->ctrl_blk);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200273 break;
274 case SUM:
275 break;
276 default:
Sudip Mukherjee0cae90a2014-09-29 14:33:26 +0530277 dev_err(((struct hw *)mgr->hw)->card->dev,
278 "Invalid resource type value %d!\n",
Sudip Mukherjee62afa852014-08-26 19:01:43 +0530279 mgr->type);
Wai Yew CHAY8cc72362009-05-14 08:05:58 +0200280 break;
281 }
282
283 mgr->hw = mgr->ctrl_blk = NULL;
284 }
285
286 mgr->type = NUM_RSCTYP;
287 mgr->avail = mgr->amount = 0;
288
289 return 0;
290}