blob: 6e2cc3d571617fd19d786728342d7a5fb1dd60c2 [file] [log] [blame]
Steven Toth8d4f9d02008-05-22 18:05:26 -03001/*
Michael Krufky85447062008-05-22 18:29:20 -03002 * Siano core API module
Steven Toth8d4f9d02008-05-22 18:05:26 -03003 *
Michael Krufky85447062008-05-22 18:29:20 -03004 * This file contains implementation for the interface to sms core component
5 *
6 * author: Anatoly Greenblat
7 *
8 * Copyright (c), 2005-2008 Siano Mobile Silicon, Inc.
Steven Toth8d4f9d02008-05-22 18:05:26 -03009 *
10 * This program is free software; you can redistribute it and/or modify
Michael Krufky85447062008-05-22 18:29:20 -030011 * it under the terms of the GNU General Public License version 3 as
12 * published by the Free Software Foundation;
Steven Toth8d4f9d02008-05-22 18:05:26 -030013 *
Michael Krufky85447062008-05-22 18:29:20 -030014 * Software distributed under the License is distributed on an "AS IS"
15 * basis, WITHOUT WARRANTY OF ANY KIND, either express or implied.
Steven Toth8d4f9d02008-05-22 18:05:26 -030016 *
Michael Krufky85447062008-05-22 18:29:20 -030017 * See the GNU General Public License for more details.
Steven Toth8d4f9d02008-05-22 18:05:26 -030018 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 */
23
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030024#include <linux/kernel.h>
25#include <linux/init.h>
26#include <linux/module.h>
27#include <linux/moduleparam.h>
28#include <linux/dma-mapping.h>
29#include <linux/delay.h>
30#include <asm/io.h>
31
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030032#include <linux/firmware.h>
33
34#include "smscoreapi.h"
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030035
Michael Krufky82237412008-06-15 15:14:13 -030036#define PERROR(fmt, args...)\
Michael Krufky068d6c02008-06-19 01:15:46 -030037 sms_err("smscore error: line %d- %s(): " fmt, \
Michael Krufky82237412008-06-15 15:14:13 -030038 __LINE__, __func__, ## args)
Michael Krufkyf17407a2008-06-14 00:43:26 -030039
40#ifdef SMSCORE_DEBUG
Michael Krufkyf17407a2008-06-14 00:43:26 -030041#undef PWARNING
Michael Krufky068d6c02008-06-19 01:15:46 -030042# define PWARNING(fmt, args...) sms_info("smscore warning: " \
Michael Krufky82237412008-06-15 15:14:13 -030043 "line %d- %s(): " fmt, \
44 __LINE__, __func__, ## args)
Michael Krufkyf17407a2008-06-14 00:43:26 -030045#undef PDEBUG /* undef it, just in case */
Michael Krufky068d6c02008-06-19 01:15:46 -030046# define PDEBUG(fmt, args...) sms_info("smscore - %s(): " fmt, \
Michael Krufky82237412008-06-15 15:14:13 -030047 __func__, ## args)
Michael Krufkyf17407a2008-06-14 00:43:26 -030048#else /*SMSCORE_DEBUG*/
Michael Krufkyf17407a2008-06-14 00:43:26 -030049#define PDEBUG(fmt, args...)
50#define PWARNING(fmt, args...)
Michael Krufkyf17407a2008-06-14 00:43:26 -030051#endif
52
Michael Krufky18245e12008-06-15 17:52:24 -030053struct smscore_device_notifyee_t {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030054 struct list_head entry;
55 hotplug_t hotplug;
Michael Krufky18245e12008-06-15 17:52:24 -030056};
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030057
Michael Krufky18245e12008-06-15 17:52:24 -030058struct smscore_idlist_t {
Michael Krufkyf17407a2008-06-14 00:43:26 -030059 struct list_head entry;
60 int id;
61 int data_type;
Michael Krufky18245e12008-06-15 17:52:24 -030062};
Michael Krufkyf17407a2008-06-14 00:43:26 -030063
Michael Krufky18245e12008-06-15 17:52:24 -030064struct smscore_client_t {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030065 struct list_head entry;
Michael Krufky18245e12008-06-15 17:52:24 -030066 struct smscore_device_t *coredev;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030067 void *context;
Michael Krufkyf17407a2008-06-14 00:43:26 -030068 struct list_head idlist;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030069 onresponse_t onresponse_handler;
70 onremove_t onremove_handler;
Michael Krufky18245e12008-06-15 17:52:24 -030071};
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030072
Michael Krufky18245e12008-06-15 17:52:24 -030073struct smscore_device_t {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -030074 struct list_head entry;
75
76 struct list_head clients;
77 struct list_head subclients;
78 spinlock_t clientslock;
79
80 struct list_head buffers;
81 spinlock_t bufferslock;
82 int num_buffers;
83
84 void *common_buffer;
85 int common_buffer_size;
86 dma_addr_t common_buffer_phys;
87
88 void *context;
89 struct device *device;
90
91 char devpath[32];
92 unsigned long device_flags;
93
94 setmode_t setmode_handler;
95 detectmode_t detectmode_handler;
96 sendrequest_t sendrequest_handler;
97 preload_t preload_handler;
98 postload_t postload_handler;
99
100 int mode, modes_supported;
101
102 struct completion version_ex_done, data_download_done, trigger_done;
103 struct completion init_device_done, reload_start_done, resume_done;
Michael Krufky1c11d542008-06-18 22:09:55 -0300104
105 int board_id;
Michael Krufky18245e12008-06-15 17:52:24 -0300106};
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300107
Michael Krufky1c11d542008-06-18 22:09:55 -0300108void smscore_set_board_id(struct smscore_device_t *core, int id)
109{
110 core->board_id = id;
111}
112
113int smscore_get_board_id(struct smscore_device_t *core)
114{
115 return core->board_id;
116}
117
Michael Krufky18245e12008-06-15 17:52:24 -0300118struct smscore_registry_entry_t {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300119 struct list_head entry;
120 char devpath[32];
121 int mode;
Michael Krufky18245e12008-06-15 17:52:24 -0300122 enum sms_device_type_st type;
123};
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300124
125struct list_head g_smscore_notifyees;
126struct list_head g_smscore_devices;
127kmutex_t g_smscore_deviceslock;
128
129struct list_head g_smscore_registry;
130kmutex_t g_smscore_registrylock;
131
Michael Krufkydd5b2a52008-06-18 20:25:25 -0300132static int default_mode = 4;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300133
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300134module_param(default_mode, int, 0644);
135MODULE_PARM_DESC(default_mode, "default firmware id (device mode)");
136
Michael Krufky18245e12008-06-15 17:52:24 -0300137static struct smscore_registry_entry_t *smscore_find_registry(char *devpath)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300138{
Michael Krufky18245e12008-06-15 17:52:24 -0300139 struct smscore_registry_entry_t *entry;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300140 struct list_head *next;
141
142 kmutex_lock(&g_smscore_registrylock);
Michael Krufky82237412008-06-15 15:14:13 -0300143 for (next = g_smscore_registry.next;
144 next != &g_smscore_registry;
145 next = next->next) {
Michael Krufky18245e12008-06-15 17:52:24 -0300146 entry = (struct smscore_registry_entry_t *) next;
Michael Krufky82237412008-06-15 15:14:13 -0300147 if (!strcmp(entry->devpath, devpath)) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300148 kmutex_unlock(&g_smscore_registrylock);
Michael Krufkyf17407a2008-06-14 00:43:26 -0300149 return entry;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300150 }
151 }
Michael Krufky18245e12008-06-15 17:52:24 -0300152 entry = (struct smscore_registry_entry_t *)
153 kmalloc(sizeof(struct smscore_registry_entry_t),
154 GFP_KERNEL);
Michael Krufky82237412008-06-15 15:14:13 -0300155 if (entry) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300156 entry->mode = default_mode;
157 strcpy(entry->devpath, devpath);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300158 list_add(&entry->entry, &g_smscore_registry);
Michael Krufky82237412008-06-15 15:14:13 -0300159 } else
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300160 sms_err("failed to create smscore_registry.");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300161 kmutex_unlock(&g_smscore_registrylock);
Michael Krufkyf17407a2008-06-14 00:43:26 -0300162 return entry;
163}
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300164
Michael Krufky82237412008-06-15 15:14:13 -0300165int smscore_registry_getmode(char *devpath)
Michael Krufkyf17407a2008-06-14 00:43:26 -0300166{
Michael Krufky18245e12008-06-15 17:52:24 -0300167 struct smscore_registry_entry_t *entry;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300168
Michael Krufky82237412008-06-15 15:14:13 -0300169 entry = smscore_find_registry(devpath);
170 if (entry)
Michael Krufkyf17407a2008-06-14 00:43:26 -0300171 return entry->mode;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300172 else
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300173 sms_err("No registry found.");
Michael Krufky82237412008-06-15 15:14:13 -0300174
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300175 return default_mode;
176}
177
Michael Krufky18245e12008-06-15 17:52:24 -0300178enum sms_device_type_st smscore_registry_gettype(char *devpath)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300179{
Michael Krufky18245e12008-06-15 17:52:24 -0300180 struct smscore_registry_entry_t *entry;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300181
Michael Krufky82237412008-06-15 15:14:13 -0300182 entry = smscore_find_registry(devpath);
183 if (entry)
Michael Krufkyf17407a2008-06-14 00:43:26 -0300184 return entry->type;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300185 else
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300186 sms_err("No registry found.");
Michael Krufky82237412008-06-15 15:14:13 -0300187
Michael Krufkyf17407a2008-06-14 00:43:26 -0300188 return -1;
189}
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300190
Michael Krufky82237412008-06-15 15:14:13 -0300191void smscore_registry_setmode(char *devpath, int mode)
Michael Krufkyf17407a2008-06-14 00:43:26 -0300192{
Michael Krufky18245e12008-06-15 17:52:24 -0300193 struct smscore_registry_entry_t *entry;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300194
Michael Krufky82237412008-06-15 15:14:13 -0300195 entry = smscore_find_registry(devpath);
196 if (entry)
197 entry->mode = mode;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300198 else
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300199 sms_err("No registry found.");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300200}
Michael Krufky82237412008-06-15 15:14:13 -0300201
Michael Krufky18245e12008-06-15 17:52:24 -0300202void smscore_registry_settype(char *devpath, enum sms_device_type_st type)
Michael Krufky82237412008-06-15 15:14:13 -0300203{
Michael Krufky18245e12008-06-15 17:52:24 -0300204 struct smscore_registry_entry_t *entry;
Michael Krufky82237412008-06-15 15:14:13 -0300205
206 entry = smscore_find_registry(devpath);
207 if (entry)
208 entry->type = type;
209 else
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300210 sms_err("No registry found.");
Michael Krufkyf17407a2008-06-14 00:43:26 -0300211}
212
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300213
Michael Krufkya83ccdd2008-05-06 03:11:51 -0300214void list_add_locked(struct list_head *new, struct list_head *head,
Michael Krufky82237412008-06-15 15:14:13 -0300215 spinlock_t *lock)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300216{
217 unsigned long flags;
218
219 spin_lock_irqsave(lock, flags);
220
221 list_add(new, head);
222
223 spin_unlock_irqrestore(lock, flags);
224}
225
226/**
227 * register a client callback that called when device plugged in/unplugged
228 * NOTE: if devices exist callback is called immediately for each device
229 *
230 * @param hotplug callback
231 *
232 * @return 0 on success, <0 on error.
233 */
234int smscore_register_hotplug(hotplug_t hotplug)
235{
Michael Krufky18245e12008-06-15 17:52:24 -0300236 struct smscore_device_notifyee_t *notifyee;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300237 struct list_head *next, *first;
238 int rc = 0;
239
240 kmutex_lock(&g_smscore_deviceslock);
241
Michael Krufky18245e12008-06-15 17:52:24 -0300242 notifyee = kmalloc(sizeof(struct smscore_device_notifyee_t),
243 GFP_KERNEL);
Michael Krufky82237412008-06-15 15:14:13 -0300244 if (notifyee) {
245 /* now notify callback about existing devices */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300246 first = &g_smscore_devices;
Michael Krufky82237412008-06-15 15:14:13 -0300247 for (next = first->next;
248 next != first && !rc;
249 next = next->next) {
Michael Krufky18245e12008-06-15 17:52:24 -0300250 struct smscore_device_t *coredev =
251 (struct smscore_device_t *) next;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300252 rc = hotplug(coredev, coredev->device, 1);
253 }
254
Michael Krufky82237412008-06-15 15:14:13 -0300255 if (rc >= 0) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300256 notifyee->hotplug = hotplug;
257 list_add(&notifyee->entry, &g_smscore_notifyees);
Michael Krufky82237412008-06-15 15:14:13 -0300258 } else
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300259 kfree(notifyee);
Michael Krufky82237412008-06-15 15:14:13 -0300260 } else
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300261 rc = -ENOMEM;
262
263 kmutex_unlock(&g_smscore_deviceslock);
264
265 return rc;
266}
267
268/**
269 * unregister a client callback that called when device plugged in/unplugged
270 *
271 * @param hotplug callback
272 *
273 */
274void smscore_unregister_hotplug(hotplug_t hotplug)
275{
276 struct list_head *next, *first;
277
278 kmutex_lock(&g_smscore_deviceslock);
279
280 first = &g_smscore_notifyees;
281
Michael Krufky82237412008-06-15 15:14:13 -0300282 for (next = first->next; next != first;) {
Michael Krufky18245e12008-06-15 17:52:24 -0300283 struct smscore_device_notifyee_t *notifyee =
284 (struct smscore_device_notifyee_t *) next;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300285 next = next->next;
286
Michael Krufky82237412008-06-15 15:14:13 -0300287 if (notifyee->hotplug == hotplug) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300288 list_del(&notifyee->entry);
289 kfree(notifyee);
290 }
291 }
292
293 kmutex_unlock(&g_smscore_deviceslock);
294}
295
Michael Krufky18245e12008-06-15 17:52:24 -0300296void smscore_notify_clients(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300297{
Michael Krufky18245e12008-06-15 17:52:24 -0300298 struct smscore_client_t *client;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300299
Michael Krufky82237412008-06-15 15:14:13 -0300300 /* the client must call smscore_unregister_client from remove handler */
301 while (!list_empty(&coredev->clients)) {
Michael Krufky18245e12008-06-15 17:52:24 -0300302 client = (struct smscore_client_t *) coredev->clients.next;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300303 client->onremove_handler(client->context);
304 }
305}
306
Michael Krufky18245e12008-06-15 17:52:24 -0300307int smscore_notify_callbacks(struct smscore_device_t *coredev,
308 struct device *device, int arrival)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300309{
310 struct list_head *next, *first;
311 int rc = 0;
312
Michael Krufky82237412008-06-15 15:14:13 -0300313 /* note: must be called under g_deviceslock */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300314
315 first = &g_smscore_notifyees;
316
Michael Krufky82237412008-06-15 15:14:13 -0300317 for (next = first->next; next != first; next = next->next) {
Michael Krufky18245e12008-06-15 17:52:24 -0300318 rc = ((struct smscore_device_notifyee_t *) next)->
Michael Krufky59bf6b82008-06-15 16:50:11 -0300319 hotplug(coredev, device, arrival);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300320 if (rc < 0)
321 break;
322 }
323
324 return rc;
325}
326
Michael Krufky18245e12008-06-15 17:52:24 -0300327struct smscore_buffer_t *smscore_createbuffer(u8 *buffer, void *common_buffer,
Michael Krufkya83ccdd2008-05-06 03:11:51 -0300328 dma_addr_t common_buffer_phys)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300329{
Michael Krufky18245e12008-06-15 17:52:24 -0300330 struct smscore_buffer_t *cb =
331 kmalloc(sizeof(struct smscore_buffer_t), GFP_KERNEL);
Michael Krufky82237412008-06-15 15:14:13 -0300332 if (!cb) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300333 sms_info("kmalloc(...) failed");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300334 return NULL;
335 }
336
337 cb->p = buffer;
Michael Krufky494d24c2008-06-14 07:40:41 -0300338 cb->offset_in_common = buffer - (u8 *) common_buffer;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300339 cb->phys = common_buffer_phys + cb->offset_in_common;
340
341 return cb;
342}
343
344/**
Michael Krufky82237412008-06-15 15:14:13 -0300345 * creates coredev object for a device, prepares buffers,
346 * creates buffer mappings, notifies registered hotplugs about new device.
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300347 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300348 * @param params device pointer to struct with device specific parameters
349 * and handlers
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300350 * @param coredev pointer to a value that receives created coredev object
351 *
352 * @return 0 on success, <0 on error.
353 */
Michael Krufky18245e12008-06-15 17:52:24 -0300354int smscore_register_device(struct smsdevice_params_t *params,
355 struct smscore_device_t **coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300356{
Michael Krufky18245e12008-06-15 17:52:24 -0300357 struct smscore_device_t *dev;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300358 u8 *buffer;
359
Michael Krufky18245e12008-06-15 17:52:24 -0300360 dev = kzalloc(sizeof(struct smscore_device_t), GFP_KERNEL);
Michael Krufky82237412008-06-15 15:14:13 -0300361 if (!dev) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300362 sms_info("kzalloc(...) failed");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300363 return -ENOMEM;
364 }
365
Michael Krufky82237412008-06-15 15:14:13 -0300366 /* init list entry so it could be safe in smscore_unregister_device */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300367 INIT_LIST_HEAD(&dev->entry);
368
Michael Krufky82237412008-06-15 15:14:13 -0300369 /* init queues */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300370 INIT_LIST_HEAD(&dev->clients);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300371 INIT_LIST_HEAD(&dev->buffers);
372
Michael Krufky82237412008-06-15 15:14:13 -0300373 /* init locks */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300374 spin_lock_init(&dev->clientslock);
375 spin_lock_init(&dev->bufferslock);
376
Michael Krufky82237412008-06-15 15:14:13 -0300377 /* init completion events */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300378 init_completion(&dev->version_ex_done);
379 init_completion(&dev->data_download_done);
380 init_completion(&dev->trigger_done);
381 init_completion(&dev->init_device_done);
382 init_completion(&dev->reload_start_done);
383 init_completion(&dev->resume_done);
384
Michael Krufky82237412008-06-15 15:14:13 -0300385 /* alloc common buffer */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300386 dev->common_buffer_size = params->buffer_size * params->num_buffers;
Michael Krufky82237412008-06-15 15:14:13 -0300387 dev->common_buffer = dma_alloc_coherent(NULL, dev->common_buffer_size,
388 &dev->common_buffer_phys,
389 GFP_KERNEL | GFP_DMA);
390 if (!dev->common_buffer) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300391 smscore_unregister_device(dev);
392 return -ENOMEM;
393 }
394
Michael Krufky82237412008-06-15 15:14:13 -0300395 /* prepare dma buffers */
396 for (buffer = dev->common_buffer;
397 dev->num_buffers < params->num_buffers;
Michael Krufkyfa830e82008-06-15 15:52:43 -0300398 dev->num_buffers++, buffer += params->buffer_size) {
Michael Krufky18245e12008-06-15 17:52:24 -0300399 struct smscore_buffer_t *cb =
Michael Krufky59bf6b82008-06-15 16:50:11 -0300400 smscore_createbuffer(buffer, dev->common_buffer,
401 dev->common_buffer_phys);
Michael Krufky82237412008-06-15 15:14:13 -0300402 if (!cb) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300403 smscore_unregister_device(dev);
404 return -ENOMEM;
405 }
406
407 smscore_putbuffer(dev, cb);
408 }
409
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300410 sms_info("allocated %d buffers", dev->num_buffers);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300411
412 dev->mode = DEVICE_MODE_NONE;
413 dev->context = params->context;
414 dev->device = params->device;
415 dev->setmode_handler = params->setmode_handler;
416 dev->detectmode_handler = params->detectmode_handler;
417 dev->sendrequest_handler = params->sendrequest_handler;
418 dev->preload_handler = params->preload_handler;
419 dev->postload_handler = params->postload_handler;
420
421 dev->device_flags = params->flags;
422 strcpy(dev->devpath, params->devpath);
423
Michael Krufky82237412008-06-15 15:14:13 -0300424 smscore_registry_settype(dev->devpath, params->device_type);
Michael Krufkyf17407a2008-06-14 00:43:26 -0300425
Michael Krufky82237412008-06-15 15:14:13 -0300426 /* add device to devices list */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300427 kmutex_lock(&g_smscore_deviceslock);
428 list_add(&dev->entry, &g_smscore_devices);
429 kmutex_unlock(&g_smscore_deviceslock);
430
431 *coredev = dev;
432
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300433 sms_info("device %p created", dev);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300434
435 return 0;
436}
437
438/**
439 * sets initial device mode and notifies client hotplugs that device is ready
440 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300441 * @param coredev pointer to a coredev object returned by
442 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300443 *
444 * @return 0 on success, <0 on error.
445 */
Michael Krufky18245e12008-06-15 17:52:24 -0300446int smscore_start_device(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300447{
Michael Krufky59bf6b82008-06-15 16:50:11 -0300448 int rc = smscore_set_device_mode(
449 coredev, smscore_registry_getmode(coredev->devpath));
Michael Krufky82237412008-06-15 15:14:13 -0300450 if (rc < 0) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300451 sms_info("set device mode faile , rc %d", rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300452 return rc;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300453 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300454
455 kmutex_lock(&g_smscore_deviceslock);
456
457 rc = smscore_notify_callbacks(coredev, coredev->device, 1);
458
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300459 sms_info("device %p started, rc %d", coredev, rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300460
461 kmutex_unlock(&g_smscore_deviceslock);
462
463 return rc;
464}
465
Michael Krufky18245e12008-06-15 17:52:24 -0300466int smscore_sendrequest_and_wait(struct smscore_device_t *coredev, void *buffer,
Michael Krufky82237412008-06-15 15:14:13 -0300467 size_t size, struct completion *completion)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300468{
469 int rc = coredev->sendrequest_handler(coredev->context, buffer, size);
Michael Krufky82237412008-06-15 15:14:13 -0300470 if (rc < 0) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300471 sms_info("sendrequest returned error %d", rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300472 return rc;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300473 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300474
Michael Krufky82237412008-06-15 15:14:13 -0300475 return wait_for_completion_timeout(completion,
476 msecs_to_jiffies(10000)) ?
477 0 : -ETIME;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300478}
479
Michael Krufky18245e12008-06-15 17:52:24 -0300480int smscore_load_firmware_family2(struct smscore_device_t *coredev,
481 void *buffer, size_t size)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300482{
Michael Krufky18245e12008-06-15 17:52:24 -0300483 struct SmsFirmware_ST *firmware = (struct SmsFirmware_ST *) buffer;
484 struct SmsMsgHdr_ST *msg;
Michael Krufkyf0333e32008-06-15 15:32:00 -0300485 u32 mem_address = firmware->StartAddress;
Michael Krufkya83ccdd2008-05-06 03:11:51 -0300486 u8 *payload = firmware->Payload;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300487 int rc = 0;
488
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300489 sms_info("loading FW to addr 0x%x size %d",
490 mem_address, firmware->Length);
Michael Krufky82237412008-06-15 15:14:13 -0300491 if (coredev->preload_handler) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300492 rc = coredev->preload_handler(coredev->context);
493 if (rc < 0)
494 return rc;
495 }
496
Michael Krufky82237412008-06-15 15:14:13 -0300497 /* PAGE_SIZE buffer shall be enough and dma aligned */
Michael Krufkye0808422008-06-15 19:11:37 -0300498 msg = kmalloc(PAGE_SIZE, GFP_KERNEL | GFP_DMA);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300499 if (!msg)
500 return -ENOMEM;
501
Michael Krufky82237412008-06-15 15:14:13 -0300502 if (coredev->mode != DEVICE_MODE_NONE) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300503 PDEBUG("Sending reload command");
Michael Krufky82237412008-06-15 15:14:13 -0300504 SMS_INIT_MSG(msg, MSG_SW_RELOAD_START_REQ,
Michael Krufky18245e12008-06-15 17:52:24 -0300505 sizeof(struct SmsMsgHdr_ST));
Michael Krufky82237412008-06-15 15:14:13 -0300506 rc = smscore_sendrequest_and_wait(coredev, msg,
507 msg->msgLength,
508 &coredev->reload_start_done);
Michael Krufkyf0333e32008-06-15 15:32:00 -0300509 mem_address = *(u32 *) &payload[20];
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300510 }
511
Michael Krufkyfa830e82008-06-15 15:52:43 -0300512 while (size && rc >= 0) {
Michael Krufky18245e12008-06-15 17:52:24 -0300513 struct SmsDataDownload_ST *DataMsg =
514 (struct SmsDataDownload_ST *) msg;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300515 int payload_size = min((int) size, SMS_MAX_PAYLOAD_SIZE);
516
Michael Krufky82237412008-06-15 15:14:13 -0300517 SMS_INIT_MSG(msg, MSG_SMS_DATA_DOWNLOAD_REQ,
Michael Krufky18245e12008-06-15 17:52:24 -0300518 (u16)(sizeof(struct SmsMsgHdr_ST) +
Michael Krufkyf0333e32008-06-15 15:32:00 -0300519 sizeof(u32) + payload_size));
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300520
521 DataMsg->MemAddr = mem_address;
522 memcpy(DataMsg->Payload, payload, payload_size);
523
Michael Krufky82237412008-06-15 15:14:13 -0300524 if ((coredev->device_flags & SMS_ROM_NO_RESPONSE) &&
525 (coredev->mode == DEVICE_MODE_NONE))
Michael Krufky59bf6b82008-06-15 16:50:11 -0300526 rc = coredev->sendrequest_handler(
527 coredev->context, DataMsg,
528 DataMsg->xMsgHeader.msgLength);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300529 else
Michael Krufky59bf6b82008-06-15 16:50:11 -0300530 rc = smscore_sendrequest_and_wait(
531 coredev, DataMsg,
532 DataMsg->xMsgHeader.msgLength,
533 &coredev->data_download_done);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300534
535 payload += payload_size;
536 size -= payload_size;
537 mem_address += payload_size;
538 }
539
Michael Krufky82237412008-06-15 15:14:13 -0300540 if (rc >= 0) {
541 if (coredev->mode == DEVICE_MODE_NONE) {
Michael Krufky18245e12008-06-15 17:52:24 -0300542 struct SmsMsgData_ST *TriggerMsg =
543 (struct SmsMsgData_ST *) msg;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300544
Michael Krufky82237412008-06-15 15:14:13 -0300545 SMS_INIT_MSG(msg, MSG_SMS_SWDOWNLOAD_TRIGGER_REQ,
Michael Krufky18245e12008-06-15 17:52:24 -0300546 sizeof(struct SmsMsgHdr_ST) +
Michael Krufkyf0333e32008-06-15 15:32:00 -0300547 sizeof(u32) * 5);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300548
Michael Krufky59bf6b82008-06-15 16:50:11 -0300549 TriggerMsg->msgData[0] = firmware->StartAddress;
550 /* Entry point */
Michael Krufkyfa830e82008-06-15 15:52:43 -0300551 TriggerMsg->msgData[1] = 5; /* Priority */
552 TriggerMsg->msgData[2] = 0x200; /* Stack size */
553 TriggerMsg->msgData[3] = 0; /* Parameter */
554 TriggerMsg->msgData[4] = 4; /* Task ID */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300555
Michael Krufky82237412008-06-15 15:14:13 -0300556 if (coredev->device_flags & SMS_ROM_NO_RESPONSE) {
Michael Krufky59bf6b82008-06-15 16:50:11 -0300557 rc = coredev->sendrequest_handler(
558 coredev->context, TriggerMsg,
559 TriggerMsg->xMsgHeader.msgLength);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300560 msleep(100);
Michael Krufky82237412008-06-15 15:14:13 -0300561 } else
Michael Krufky59bf6b82008-06-15 16:50:11 -0300562 rc = smscore_sendrequest_and_wait(
563 coredev, TriggerMsg,
564 TriggerMsg->xMsgHeader.msgLength,
565 &coredev->trigger_done);
Michael Krufky82237412008-06-15 15:14:13 -0300566 } else {
567 SMS_INIT_MSG(msg, MSG_SW_RELOAD_EXEC_REQ,
Michael Krufky18245e12008-06-15 17:52:24 -0300568 sizeof(struct SmsMsgHdr_ST));
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300569
Michael Krufky82237412008-06-15 15:14:13 -0300570 rc = coredev->sendrequest_handler(coredev->context,
571 msg, msg->msgLength);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300572 }
Michael Krufky82237412008-06-15 15:14:13 -0300573 msleep(500);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300574 }
575
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300576 sms_debug("rc=%d, postload=%p ", rc,
Michael Krufky068d6c02008-06-19 01:15:46 -0300577 coredev->postload_handler);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300578
579 kfree(msg);
580
Michael Krufkyf17407a2008-06-14 00:43:26 -0300581 return ((rc >= 0) && coredev->postload_handler) ?
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300582 coredev->postload_handler(coredev->context) :
583 rc;
584}
585
586/**
587 * loads specified firmware into a buffer and calls device loadfirmware_handler
588 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300589 * @param coredev pointer to a coredev object returned by
590 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300591 * @param filename null-terminated string specifies firmware file name
592 * @param loadfirmware_handler device handler that loads firmware
593 *
594 * @return 0 on success, <0 on error.
595 */
Michael Krufky18245e12008-06-15 17:52:24 -0300596int smscore_load_firmware_from_file(struct smscore_device_t *coredev,
597 char *filename,
Michael Krufky82237412008-06-15 15:14:13 -0300598 loadfirmware_t loadfirmware_handler)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300599{
600 int rc = -ENOENT;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300601 const struct firmware *fw;
Michael Krufkya83ccdd2008-05-06 03:11:51 -0300602 u8 *fw_buffer;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300603
Michael Krufky82237412008-06-15 15:14:13 -0300604 if (loadfirmware_handler == NULL && !(coredev->device_flags &
605 SMS_DEVICE_FAMILY2))
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300606 return -EINVAL;
607
608 rc = request_firmware(&fw, filename, coredev->device);
Michael Krufky82237412008-06-15 15:14:13 -0300609 if (rc < 0) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300610 sms_info("failed to open \"%s\"", filename);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300611 return rc;
612 }
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300613 sms_info("read FW %s, size=%d\"", filename, fw->size);
Michael Krufky82237412008-06-15 15:14:13 -0300614 fw_buffer = kmalloc(ALIGN(fw->size, SMS_ALLOC_ALIGNMENT),
615 GFP_KERNEL | GFP_DMA);
616 if (fw_buffer) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300617 memcpy(fw_buffer, fw->data, fw->size);
618
619 rc = (coredev->device_flags & SMS_DEVICE_FAMILY2) ?
Michael Krufky59bf6b82008-06-15 16:50:11 -0300620 smscore_load_firmware_family2(coredev,
621 fw_buffer,
622 fw->size) :
623 loadfirmware_handler(coredev->context,
624 fw_buffer, fw->size);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300625
626 kfree(fw_buffer);
Michael Krufky82237412008-06-15 15:14:13 -0300627 } else {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300628 sms_info("failed to allocate firmware buffer");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300629 rc = -ENOMEM;
630 }
631
632 release_firmware(fw);
633
634 return rc;
635}
636
Michael Krufky18245e12008-06-15 17:52:24 -0300637int smscore_load_firmware_from_buffer(struct smscore_device_t *coredev,
638 u8 *buffer, int size, int new_mode)
Michael Krufkyf17407a2008-06-14 00:43:26 -0300639{
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300640 PERROR("Feature not implemented yet");
Michael Krufkyf17407a2008-06-14 00:43:26 -0300641 return -EFAULT;
642}
643
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300644/**
Michael Krufky59bf6b82008-06-15 16:50:11 -0300645 * notifies all clients registered with the device, notifies hotplugs,
646 * frees all buffers and coredev object
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300647 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300648 * @param coredev pointer to a coredev object returned by
649 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300650 *
651 * @return 0 on success, <0 on error.
652 */
Michael Krufky18245e12008-06-15 17:52:24 -0300653void smscore_unregister_device(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300654{
Michael Krufky18245e12008-06-15 17:52:24 -0300655 struct smscore_buffer_t *cb;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300656 int num_buffers = 0;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300657 int retry = 0;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300658
659 kmutex_lock(&g_smscore_deviceslock);
660
661 smscore_notify_clients(coredev);
662 smscore_notify_callbacks(coredev, NULL, 0);
663
Michael Krufky82237412008-06-15 15:14:13 -0300664 /* at this point all buffers should be back
665 * onresponse must no longer be called */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300666
Michael Krufky82237412008-06-15 15:14:13 -0300667 while (1) {
668 while ((cb = smscore_getbuffer(coredev))) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300669 kfree(cb);
Michael Krufkyfa830e82008-06-15 15:52:43 -0300670 num_buffers++;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300671 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300672 if (num_buffers == coredev->num_buffers)
673 break;
Michael Krufky82237412008-06-15 15:14:13 -0300674 if (++retry > 10) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300675 sms_info("exiting although "
676 "not all buffers released.");
Michael Krufkyf17407a2008-06-14 00:43:26 -0300677 break;
678 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300679
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300680 sms_info("waiting for %d buffer(s)",
Michael Krufky068d6c02008-06-19 01:15:46 -0300681 coredev->num_buffers - num_buffers);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300682 msleep(100);
683 }
684
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300685 sms_info("freed %d buffers", num_buffers);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300686
687 if (coredev->common_buffer)
Michael Krufky82237412008-06-15 15:14:13 -0300688 dma_free_coherent(NULL, coredev->common_buffer_size,
689 coredev->common_buffer,
690 coredev->common_buffer_phys);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300691
692 list_del(&coredev->entry);
693 kfree(coredev);
694
695 kmutex_unlock(&g_smscore_deviceslock);
696
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300697 sms_info("device %p destroyed", coredev);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300698}
699
Michael Krufky18245e12008-06-15 17:52:24 -0300700int smscore_detect_mode(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300701{
Michael Krufky18245e12008-06-15 17:52:24 -0300702 void *buffer = kmalloc(sizeof(struct SmsMsgHdr_ST) + SMS_DMA_ALIGNMENT,
Michael Krufky82237412008-06-15 15:14:13 -0300703 GFP_KERNEL | GFP_DMA);
Michael Krufky18245e12008-06-15 17:52:24 -0300704 struct SmsMsgHdr_ST *msg =
705 (struct SmsMsgHdr_ST *) SMS_ALIGN_ADDRESS(buffer);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300706 int rc;
707
708 if (!buffer)
709 return -ENOMEM;
710
Michael Krufky18245e12008-06-15 17:52:24 -0300711 SMS_INIT_MSG(msg, MSG_SMS_GET_VERSION_EX_REQ,
712 sizeof(struct SmsMsgHdr_ST));
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300713
Michael Krufky82237412008-06-15 15:14:13 -0300714 rc = smscore_sendrequest_and_wait(coredev, msg, msg->msgLength,
715 &coredev->version_ex_done);
716 if (rc == -ETIME) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300717 sms_err("MSG_SMS_GET_VERSION_EX_REQ failed first try");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300718
Michael Krufky82237412008-06-15 15:14:13 -0300719 if (wait_for_completion_timeout(&coredev->resume_done,
720 msecs_to_jiffies(5000))) {
Michael Krufky59bf6b82008-06-15 16:50:11 -0300721 rc = smscore_sendrequest_and_wait(
722 coredev, msg, msg->msgLength,
723 &coredev->version_ex_done);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300724 if (rc < 0)
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300725 sms_err("MSG_SMS_GET_VERSION_EX_REQ failed "
726 "second try, rc %d", rc);
Michael Krufky82237412008-06-15 15:14:13 -0300727 } else
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300728 rc = -ETIME;
729 }
730
731 kfree(buffer);
732
733 return rc;
734}
735
Michael Krufky82237412008-06-15 15:14:13 -0300736char *smscore_fw_lkup[][SMS_NUM_OF_DEVICE_TYPES] = {
Michael Krufkyfbd05c82008-06-15 19:27:35 -0300737 /*Stellar NOVA A0 Nova B0 VEGA*/
738 /*DVBT*/
739 {"none", "dvb_nova_12mhz.inp", "dvb_nova_12mhz_b0.inp", "none"},
740 /*DVBH*/
741 {"none", "dvb_nova_12mhz.inp", "dvb_nova_12mhz_b0.inp", "none"},
742 /*TDMB*/
743 {"none", "tdmb_nova_12mhz.inp", "none", "none"},
744 /*DABIP*/
745 {"none", "none", "none", "none"},
746 /*BDA*/
747 {"none", "dvb_nova_12mhz.inp", "dvb_nova_12mhz_b0.inp", "none"},
748 /*ISDBT*/
749 {"none", "isdbt_nova_12mhz.inp", "dvb_nova_12mhz.inp", "none"},
750 /*ISDBTBDA*/
751 {"none", "isdbt_nova_12mhz.inp", "isdbt_nova_12mhz_b0.inp", "none"},
752 /*CMMB*/
753 {"none", "none", "none", "cmmb_vega_12mhz.inp"}
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300754};
755
Michael Krufkyf17407a2008-06-14 00:43:26 -0300756
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300757/**
758 * calls device handler to change mode of operation
759 * NOTE: stellar/usb may disconnect when changing mode
760 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300761 * @param coredev pointer to a coredev object returned by
762 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300763 * @param mode requested mode of operation
764 *
765 * @return 0 on success, <0 on error.
766 */
Michael Krufky18245e12008-06-15 17:52:24 -0300767int smscore_set_device_mode(struct smscore_device_t *coredev, int mode)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300768{
769 void *buffer;
770 int rc = 0;
Michael Krufky18245e12008-06-15 17:52:24 -0300771 enum sms_device_type_st type;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300772
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300773 PDEBUG("set device mode to %d", mode);
Michael Krufky82237412008-06-15 15:14:13 -0300774 if (coredev->device_flags & SMS_DEVICE_FAMILY2) {
775 if (mode < DEVICE_MODE_DVBT || mode > DEVICE_MODE_RAW_TUNER) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300776 sms_info("invalid mode specified %d", mode);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300777 return -EINVAL;
778 }
779
Michael Krufkyf17407a2008-06-14 00:43:26 -0300780 smscore_registry_setmode(coredev->devpath, mode);
781
Michael Krufky82237412008-06-15 15:14:13 -0300782 if (!(coredev->device_flags & SMS_DEVICE_NOT_READY)) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300783 rc = smscore_detect_mode(coredev);
Michael Krufky82237412008-06-15 15:14:13 -0300784 if (rc < 0) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300785 sms_info("mode detect failed %d", rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300786 return rc;
Michael Krufky82237412008-06-15 15:14:13 -0300787 }
Michael Krufkyf17407a2008-06-14 00:43:26 -0300788 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300789
Michael Krufky82237412008-06-15 15:14:13 -0300790 if (coredev->mode == mode) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300791 sms_info("device mode %d already set", mode);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300792 return 0;
793 }
794
Michael Krufky82237412008-06-15 15:14:13 -0300795 if (!(coredev->modes_supported & (1 << mode))) {
796 type = smscore_registry_gettype(coredev->devpath);
Michael Krufky59bf6b82008-06-15 16:50:11 -0300797 rc = smscore_load_firmware_from_file(
798 coredev, smscore_fw_lkup[mode][type], NULL);
Michael Krufky82237412008-06-15 15:14:13 -0300799 if (rc < 0) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300800 sms_info("load firmware "
801 "failed %d", rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300802 return rc;
Michael Krufky82237412008-06-15 15:14:13 -0300803 }
804 } else
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300805 sms_info("mode %d supported by running "
806 "firmware", mode);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300807
Michael Krufky18245e12008-06-15 17:52:24 -0300808 buffer = kmalloc(sizeof(struct SmsMsgData_ST) +
809 SMS_DMA_ALIGNMENT, GFP_KERNEL | GFP_DMA);
Michael Krufky82237412008-06-15 15:14:13 -0300810 if (buffer) {
Michael Krufky18245e12008-06-15 17:52:24 -0300811 struct SmsMsgData_ST *msg =
812 (struct SmsMsgData_ST *)
813 SMS_ALIGN_ADDRESS(buffer);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300814
Michael Krufky59bf6b82008-06-15 16:50:11 -0300815 SMS_INIT_MSG(&msg->xMsgHeader, MSG_SMS_INIT_DEVICE_REQ,
Michael Krufky18245e12008-06-15 17:52:24 -0300816 sizeof(struct SmsMsgData_ST));
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300817 msg->msgData[0] = mode;
818
Michael Krufky59bf6b82008-06-15 16:50:11 -0300819 rc = smscore_sendrequest_and_wait(
820 coredev, msg, msg->xMsgHeader.msgLength,
821 &coredev->init_device_done);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300822
823 kfree(buffer);
Michael Krufky82237412008-06-15 15:14:13 -0300824 } else {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300825 sms_info("Could not allocate buffer for "
826 "init device message.");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300827 rc = -ENOMEM;
Michael Krufky82237412008-06-15 15:14:13 -0300828 }
829 } else {
830 if (mode < DEVICE_MODE_DVBT || mode > DEVICE_MODE_DVBT_BDA) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300831 sms_info("invalid mode specified %d", mode);
Michael Krufkyf17407a2008-06-14 00:43:26 -0300832 return -EINVAL;
833 }
834
835 smscore_registry_setmode(coredev->devpath, mode);
836
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300837 if (coredev->detectmode_handler)
Michael Krufky82237412008-06-15 15:14:13 -0300838 coredev->detectmode_handler(coredev->context,
839 &coredev->mode);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300840
841 if (coredev->mode != mode && coredev->setmode_handler)
842 rc = coredev->setmode_handler(coredev->context, mode);
843 }
844
Michael Krufky82237412008-06-15 15:14:13 -0300845 if (rc >= 0) {
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300846 coredev->mode = mode;
847 coredev->device_flags &= ~SMS_DEVICE_NOT_READY;
848 }
849
Michael Krufkyf17407a2008-06-14 00:43:26 -0300850 if (rc != 0)
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300851 sms_info("return error code %d.", rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300852 return rc;
853}
854
855/**
856 * calls device handler to get current mode of operation
857 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300858 * @param coredev pointer to a coredev object returned by
859 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300860 *
861 * @return current mode
862 */
Michael Krufky18245e12008-06-15 17:52:24 -0300863int smscore_get_device_mode(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300864{
865 return coredev->mode;
866}
867
Michael Krufkyf17407a2008-06-14 00:43:26 -0300868/**
869 * find client by response id & type within the clients list.
870 * return client handle or NULL.
871 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300872 * @param coredev pointer to a coredev object returned by
873 * smscore_register_device
Michael Krufkyf17407a2008-06-14 00:43:26 -0300874 * @param data_type client data type (SMS_DONT_CARE for all types)
Michael Krufky82237412008-06-15 15:14:13 -0300875 * @param id client id (SMS_DONT_CARE for all id)
Michael Krufkyf17407a2008-06-14 00:43:26 -0300876 *
877 */
Michael Krufky18245e12008-06-15 17:52:24 -0300878struct smscore_client_t *smscore_find_client(struct smscore_device_t *coredev,
Michael Krufky59bf6b82008-06-15 16:50:11 -0300879 int data_type, int id)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300880{
Michael Krufky18245e12008-06-15 17:52:24 -0300881 struct smscore_client_t *client = NULL;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300882 struct list_head *next, *first;
883 unsigned long flags;
Michael Krufkyf17407a2008-06-14 00:43:26 -0300884 struct list_head *firstid, *nextid;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300885
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300886
887 spin_lock_irqsave(&coredev->clientslock, flags);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300888 first = &coredev->clients;
Michael Krufky82237412008-06-15 15:14:13 -0300889 for (next = first->next;
890 (next != first) && !client;
891 next = next->next) {
Michael Krufky18245e12008-06-15 17:52:24 -0300892 firstid = &((struct smscore_client_t *)next)->idlist;
Michael Krufky82237412008-06-15 15:14:13 -0300893 for (nextid = firstid->next;
894 nextid != firstid;
895 nextid = nextid->next) {
Michael Krufky18245e12008-06-15 17:52:24 -0300896 if ((((struct smscore_idlist_t *)nextid)->id == id) &&
897 (((struct smscore_idlist_t *)nextid)->data_type == data_type ||
898 (((struct smscore_idlist_t *)nextid)->data_type == 0))) {
899 client = (struct smscore_client_t *) next;
Michael Krufky82237412008-06-15 15:14:13 -0300900 break;
901 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300902 }
903 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300904 spin_unlock_irqrestore(&coredev->clientslock, flags);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300905 return client;
906}
907
908/**
909 * find client by response id/type, call clients onresponse handler
910 * return buffer to pool on error
911 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300912 * @param coredev pointer to a coredev object returned by
913 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300914 * @param cb pointer to response buffer descriptor
915 *
916 */
Michael Krufky18245e12008-06-15 17:52:24 -0300917void smscore_onresponse(struct smscore_device_t *coredev,
918 struct smscore_buffer_t *cb)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300919{
Michael Krufky18245e12008-06-15 17:52:24 -0300920 struct SmsMsgHdr_ST *phdr =
921 (struct SmsMsgHdr_ST *)((u8 *) cb->p + cb->offset);
922 struct smscore_client_t *client =
Michael Krufky59bf6b82008-06-15 16:50:11 -0300923 smscore_find_client(coredev, phdr->msgType, phdr->msgDstId);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300924 int rc = -EBUSY;
925
Michael Krufkyfbd05c82008-06-15 19:27:35 -0300926 static unsigned long last_sample_time; /* = 0; */
927 static int data_total; /* = 0; */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300928 unsigned long time_now = jiffies_to_msecs(jiffies);
929
930 if (!last_sample_time)
931 last_sample_time = time_now;
932
Michael Krufkyfa830e82008-06-15 15:52:43 -0300933 if (time_now - last_sample_time > 10000) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300934 sms_debug("\ndata rate %d bytes/secs",
Michael Krufky068d6c02008-06-19 01:15:46 -0300935 (int)((data_total * 1000) /
936 (time_now - last_sample_time)));
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300937
938 last_sample_time = time_now;
939 data_total = 0;
940 }
941
942 data_total += cb->size;
Michael Krufky82237412008-06-15 15:14:13 -0300943 /* If no client registered for type & id,
944 * check for control client where type is not registered */
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300945 if (client)
946 rc = client->onresponse_handler(client->context, cb);
947
Michael Krufky82237412008-06-15 15:14:13 -0300948 if (rc < 0) {
949 switch (phdr->msgType) {
950 case MSG_SMS_GET_VERSION_EX_RES:
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300951 {
Michael Krufky18245e12008-06-15 17:52:24 -0300952 struct SmsVersionRes_ST *ver =
953 (struct SmsVersionRes_ST *) phdr;
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300954 sms_debug("MSG_SMS_GET_VERSION_EX_RES "
955 "id %d prots 0x%x ver %d.%d",
Michael Krufky068d6c02008-06-19 01:15:46 -0300956 ver->FirmwareId, ver->SupportedProtocols,
957 ver->RomVersionMajor, ver->RomVersionMinor);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300958
Michael Krufky82237412008-06-15 15:14:13 -0300959 coredev->mode = ver->FirmwareId == 255 ?
960 DEVICE_MODE_NONE : ver->FirmwareId;
961 coredev->modes_supported = ver->SupportedProtocols;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300962
Michael Krufky82237412008-06-15 15:14:13 -0300963 complete(&coredev->version_ex_done);
964 break;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300965 }
Michael Krufky82237412008-06-15 15:14:13 -0300966 case MSG_SMS_INIT_DEVICE_RES:
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300967 sms_debug("MSG_SMS_INIT_DEVICE_RES");
Michael Krufky82237412008-06-15 15:14:13 -0300968 complete(&coredev->init_device_done);
969 break;
970 case MSG_SW_RELOAD_START_RES:
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300971 sms_debug("MSG_SW_RELOAD_START_RES");
Michael Krufky82237412008-06-15 15:14:13 -0300972 complete(&coredev->reload_start_done);
973 break;
974 case MSG_SMS_DATA_DOWNLOAD_RES:
975 complete(&coredev->data_download_done);
976 break;
977 case MSG_SW_RELOAD_EXEC_RES:
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300978 sms_debug("MSG_SW_RELOAD_EXEC_RES");
Michael Krufky82237412008-06-15 15:14:13 -0300979 break;
980 case MSG_SMS_SWDOWNLOAD_TRIGGER_RES:
Michael Krufkya0c0abc2008-06-19 20:35:21 -0300981 sms_debug("MSG_SMS_SWDOWNLOAD_TRIGGER_RES");
Michael Krufky82237412008-06-15 15:14:13 -0300982 complete(&coredev->trigger_done);
983 break;
984 case MSG_SMS_SLEEP_RESUME_COMP_IND:
985 complete(&coredev->resume_done);
986 break;
987 default:
988 break;
989 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300990 smscore_putbuffer(coredev, cb);
991 }
992}
993
994/**
995 * return pointer to next free buffer descriptor from core pool
996 *
Michael Krufky59bf6b82008-06-15 16:50:11 -0300997 * @param coredev pointer to a coredev object returned by
998 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -0300999 *
1000 * @return pointer to descriptor on success, NULL on error.
1001 */
Michael Krufky18245e12008-06-15 17:52:24 -03001002struct smscore_buffer_t *smscore_getbuffer(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001003{
Michael Krufky18245e12008-06-15 17:52:24 -03001004 struct smscore_buffer_t *cb = NULL;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001005 unsigned long flags;
1006
1007 spin_lock_irqsave(&coredev->bufferslock, flags);
1008
Michael Krufky82237412008-06-15 15:14:13 -03001009 if (!list_empty(&coredev->buffers)) {
Michael Krufky18245e12008-06-15 17:52:24 -03001010 cb = (struct smscore_buffer_t *) coredev->buffers.next;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001011 list_del(&cb->entry);
1012 }
1013
1014 spin_unlock_irqrestore(&coredev->bufferslock, flags);
1015
1016 return cb;
1017}
1018
1019/**
1020 * return buffer descriptor to a pool
1021 *
Michael Krufky59bf6b82008-06-15 16:50:11 -03001022 * @param coredev pointer to a coredev object returned by
1023 * smscore_register_device
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001024 * @param cb pointer buffer descriptor
1025 *
1026 */
Michael Krufky18245e12008-06-15 17:52:24 -03001027void smscore_putbuffer(struct smscore_device_t *coredev,
1028 struct smscore_buffer_t *cb)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001029{
1030 list_add_locked(&cb->entry, &coredev->buffers, &coredev->bufferslock);
1031}
1032
Michael Krufky18245e12008-06-15 17:52:24 -03001033int smscore_validate_client(struct smscore_device_t *coredev,
1034 struct smscore_client_t *client,
1035 int data_type, int id)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001036{
Michael Krufky18245e12008-06-15 17:52:24 -03001037 struct smscore_idlist_t *listentry;
1038 struct smscore_client_t *registered_client;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001039
Michael Krufky82237412008-06-15 15:14:13 -03001040 if (!client) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001041 PERROR("bad parameter.");
Michael Krufkyf17407a2008-06-14 00:43:26 -03001042 return -EFAULT;
1043 }
1044 registered_client = smscore_find_client(coredev, data_type, id);
Michael Krufkyfa830e82008-06-15 15:52:43 -03001045 if (registered_client == client)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001046 return 0;
Michael Krufkyfa830e82008-06-15 15:52:43 -03001047
Michael Krufky82237412008-06-15 15:14:13 -03001048 if (registered_client) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001049 PERROR("The msg ID already registered to another client.");
Michael Krufkyf17407a2008-06-14 00:43:26 -03001050 return -EEXIST;
1051 }
Michael Krufky18245e12008-06-15 17:52:24 -03001052 listentry = kzalloc(sizeof(struct smscore_idlist_t), GFP_KERNEL);
Michael Krufky82237412008-06-15 15:14:13 -03001053 if (!listentry) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001054 PERROR("Can't allocate memory for client id.");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001055 return -ENOMEM;
Michael Krufkyf17407a2008-06-14 00:43:26 -03001056 }
1057 listentry->id = id;
1058 listentry->data_type = data_type;
Michael Krufky82237412008-06-15 15:14:13 -03001059 list_add_locked(&listentry->entry, &client->idlist,
1060 &coredev->clientslock);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001061 return 0;
1062}
1063
1064/**
1065 * creates smsclient object, check that id is taken by another client
1066 *
1067 * @param coredev pointer to a coredev object from clients hotplug
1068 * @param initial_id all messages with this id would be sent to this client
1069 * @param data_type all messages of this type would be sent to this client
Michael Krufkyca783732008-06-15 17:36:00 -03001070 * @param onresponse_handler client handler that is called to
1071 * process incoming messages
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001072 * @param onremove_handler client handler that is called when device is removed
1073 * @param context client-specific context
1074 * @param client pointer to a value that receives created smsclient object
1075 *
1076 * @return 0 on success, <0 on error.
1077 */
Michael Krufky18245e12008-06-15 17:52:24 -03001078int smscore_register_client(struct smscore_device_t *coredev,
1079 struct smsclient_params_t *params,
1080 struct smscore_client_t **client)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001081{
Michael Krufky18245e12008-06-15 17:52:24 -03001082 struct smscore_client_t *newclient;
Michael Krufky82237412008-06-15 15:14:13 -03001083 /* check that no other channel with same parameters exists */
1084 if (smscore_find_client(coredev, params->data_type,
1085 params->initial_id)) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001086 PERROR("Client already exist.");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001087 return -EEXIST;
Michael Krufkyf17407a2008-06-14 00:43:26 -03001088 }
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001089
Michael Krufky18245e12008-06-15 17:52:24 -03001090 newclient = kzalloc(sizeof(struct smscore_client_t), GFP_KERNEL);
Michael Krufky82237412008-06-15 15:14:13 -03001091 if (!newclient) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001092 PERROR("Failed to allocate memory for client.");
Michael Krufkyf17407a2008-06-14 00:43:26 -03001093 return -ENOMEM;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001094 }
1095
Michael Krufky82237412008-06-15 15:14:13 -03001096 INIT_LIST_HEAD(&newclient->idlist);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001097 newclient->coredev = coredev;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001098 newclient->onresponse_handler = params->onresponse_handler;
1099 newclient->onremove_handler = params->onremove_handler;
1100 newclient->context = params->context;
Michael Krufky82237412008-06-15 15:14:13 -03001101 list_add_locked(&newclient->entry, &coredev->clients,
1102 &coredev->clientslock);
1103 smscore_validate_client(coredev, newclient, params->data_type,
1104 params->initial_id);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001105 *client = newclient;
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001106 PDEBUG("%p %d %d", params->context, params->data_type,
Michael Krufky82237412008-06-15 15:14:13 -03001107 params->initial_id);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001108
1109 return 0;
1110}
1111
1112/**
1113 * frees smsclient object and all subclients associated with it
1114 *
Michael Krufky59bf6b82008-06-15 16:50:11 -03001115 * @param client pointer to smsclient object returned by
1116 * smscore_register_client
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001117 *
1118 */
Michael Krufky18245e12008-06-15 17:52:24 -03001119void smscore_unregister_client(struct smscore_client_t *client)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001120{
Michael Krufky18245e12008-06-15 17:52:24 -03001121 struct smscore_device_t *coredev = client->coredev;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001122 unsigned long flags;
1123
1124 spin_lock_irqsave(&coredev->clientslock, flags);
1125
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001126
Michael Krufky82237412008-06-15 15:14:13 -03001127 while (!list_empty(&client->idlist)) {
Michael Krufky18245e12008-06-15 17:52:24 -03001128 struct smscore_idlist_t *identry =
1129 (struct smscore_idlist_t *) client->idlist.next;
Michael Krufky82237412008-06-15 15:14:13 -03001130 list_del(&identry->entry);
1131 kfree(identry);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001132 }
1133
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001134 sms_info("%p", client->context);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001135
1136 list_del(&client->entry);
1137 kfree(client);
1138
1139 spin_unlock_irqrestore(&coredev->clientslock, flags);
1140}
1141
1142/**
1143 * verifies that source id is not taken by another client,
1144 * calls device handler to send requests to the device
1145 *
Michael Krufky59bf6b82008-06-15 16:50:11 -03001146 * @param client pointer to smsclient object returned by
1147 * smscore_register_client
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001148 * @param buffer pointer to a request buffer
1149 * @param size size (in bytes) of request buffer
1150 *
1151 * @return 0 on success, <0 on error.
1152 */
Michael Krufky18245e12008-06-15 17:52:24 -03001153int smsclient_sendrequest(struct smscore_client_t *client,
1154 void *buffer, size_t size)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001155{
Michael Krufky18245e12008-06-15 17:52:24 -03001156 struct smscore_device_t *coredev;
1157 struct SmsMsgHdr_ST *phdr = (struct SmsMsgHdr_ST *) buffer;
Michael Krufkyf17407a2008-06-14 00:43:26 -03001158 int rc;
1159
Michael Krufky82237412008-06-15 15:14:13 -03001160 if (client == NULL) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001161 sms_err("Got NULL client");
Michael Krufkyf17407a2008-06-14 00:43:26 -03001162 return -EINVAL;
1163 }
1164
1165 coredev = client->coredev;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001166
Michael Krufky82237412008-06-15 15:14:13 -03001167 /* check that no other channel with same id exists */
1168 if (coredev == NULL) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001169 sms_err("Got NULL coredev");
Michael Krufkyf17407a2008-06-14 00:43:26 -03001170 return -EINVAL;
1171 }
1172
Michael Krufky82237412008-06-15 15:14:13 -03001173 rc = smscore_validate_client(client->coredev, client, 0,
1174 phdr->msgSrcId);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001175 if (rc < 0)
1176 return rc;
1177
1178 return coredev->sendrequest_handler(coredev->context, buffer, size);
1179}
1180
1181/**
1182 * return the size of large (common) buffer
1183 *
1184 * @param coredev pointer to a coredev object from clients hotplug
1185 *
1186 * @return size (in bytes) of the buffer
1187 */
Michael Krufky18245e12008-06-15 17:52:24 -03001188int smscore_get_common_buffer_size(struct smscore_device_t *coredev)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001189{
1190 return coredev->common_buffer_size;
1191}
1192
1193/**
1194 * maps common buffer (if supported by platform)
1195 *
1196 * @param coredev pointer to a coredev object from clients hotplug
1197 * @param vma pointer to vma struct from mmap handler
1198 *
1199 * @return 0 on success, <0 on error.
1200 */
Michael Krufky18245e12008-06-15 17:52:24 -03001201int smscore_map_common_buffer(struct smscore_device_t *coredev,
Michael Krufkya83ccdd2008-05-06 03:11:51 -03001202 struct vm_area_struct *vma)
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001203{
Michael Krufky82237412008-06-15 15:14:13 -03001204 unsigned long end = vma->vm_end,
1205 start = vma->vm_start,
1206 size = PAGE_ALIGN(coredev->common_buffer_size);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001207
Michael Krufky82237412008-06-15 15:14:13 -03001208 if (!(vma->vm_flags & (VM_READ | VM_SHARED)) ||
1209 (vma->vm_flags & VM_WRITE)) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001210 sms_info("invalid vm flags");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001211 return -EINVAL;
1212 }
1213
Michael Krufky82237412008-06-15 15:14:13 -03001214 if ((end - start) != size) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001215 sms_info("invalid size %d expected %d",
1216 (int)(end - start), (int) size);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001217 return -EINVAL;
1218 }
1219
Michael Krufky82237412008-06-15 15:14:13 -03001220 if (remap_pfn_range(vma, start,
1221 coredev->common_buffer_phys >> PAGE_SHIFT,
Michael Krufkydf0462e2008-06-15 19:39:55 -03001222 size, pgprot_noncached(vma->vm_page_prot))) {
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001223 sms_info("remap_page_range failed");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001224 return -EAGAIN;
1225 }
1226
1227 return 0;
1228}
1229
1230int smscore_module_init(void)
1231{
Steven Tothc6465792008-05-19 19:09:21 -03001232 int rc = 0;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001233
1234 INIT_LIST_HEAD(&g_smscore_notifyees);
1235 INIT_LIST_HEAD(&g_smscore_devices);
1236 kmutex_init(&g_smscore_deviceslock);
1237
1238 INIT_LIST_HEAD(&g_smscore_registry);
1239 kmutex_init(&g_smscore_registrylock);
1240
Steven Totheae55662008-05-22 18:04:36 -03001241 /* USB Register */
1242 rc = smsusb_register();
1243
1244 /* DVB Register */
1245 rc = smsdvb_register();
1246
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001247 sms_debug("rc %d", rc);
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001248
1249 return rc;
1250}
1251
1252void smscore_module_exit(void)
1253{
Steven Totheae55662008-05-22 18:04:36 -03001254
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001255 kmutex_lock(&g_smscore_deviceslock);
Michael Krufky82237412008-06-15 15:14:13 -03001256 while (!list_empty(&g_smscore_notifyees)) {
Michael Krufky18245e12008-06-15 17:52:24 -03001257 struct smscore_device_notifyee_t *notifyee =
1258 (struct smscore_device_notifyee_t *)
1259 g_smscore_notifyees.next;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001260
1261 list_del(&notifyee->entry);
1262 kfree(notifyee);
1263 }
1264 kmutex_unlock(&g_smscore_deviceslock);
1265
1266 kmutex_lock(&g_smscore_registrylock);
Michael Krufky82237412008-06-15 15:14:13 -03001267 while (!list_empty(&g_smscore_registry)) {
Michael Krufky18245e12008-06-15 17:52:24 -03001268 struct smscore_registry_entry_t *entry =
1269 (struct smscore_registry_entry_t *)
1270 g_smscore_registry.next;
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001271
1272 list_del(&entry->entry);
1273 kfree(entry);
1274 }
1275 kmutex_unlock(&g_smscore_registrylock);
1276
Steven Totheae55662008-05-22 18:04:36 -03001277 /* DVB UnRegister */
1278 smsdvb_unregister();
1279
1280 /* Unregister USB */
1281 smsusb_unregister();
1282
Michael Krufkya0c0abc2008-06-19 20:35:21 -03001283 sms_debug("");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001284}
1285
1286module_init(smscore_module_init);
1287module_exit(smscore_module_exit);
1288
1289MODULE_DESCRIPTION("smscore");
Michael Krufky82237412008-06-15 15:14:13 -03001290MODULE_AUTHOR("Siano Mobile Silicon,,, (doronc@siano-ms.com)");
Michael Krufky2e5c1ec82008-05-19 18:56:13 -03001291MODULE_LICENSE("GPL");