1 /******************************************************************************
3 AudioScience HPI driver
4 Copyright (C) 1997-2010 AudioScience Inc. <support@audioscience.com>
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of version 2 of the GNU General Public License as
8 published by the Free Software Foundation;
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 Extended Message Function With Response Cacheing
21 (C) Copyright AudioScience Inc. 2002
22 *****************************************************************************/
23 #define SOURCEFILE_NAME "hpimsgx.c"
24 #include "hpi_internal.h"
25 #include "hpimsginit.h"
29 static struct pci_device_id asihpi_pci_tbl
[] = {
33 static struct hpios_spinlock msgx_lock
;
35 static hpi_handler_func
*hpi_entry_points
[HPI_MAX_ADAPTERS
];
37 static hpi_handler_func
*hpi_lookup_entry_point_function(const struct hpi_pci
43 for (i
= 0; asihpi_pci_tbl
[i
].vendor
!= 0; i
++) {
44 if (asihpi_pci_tbl
[i
].vendor
!= PCI_ANY_ID
45 && asihpi_pci_tbl
[i
].vendor
!= pci_info
->vendor_id
)
47 if (asihpi_pci_tbl
[i
].device
!= PCI_ANY_ID
48 && asihpi_pci_tbl
[i
].device
!= pci_info
->device_id
)
50 if (asihpi_pci_tbl
[i
].subvendor
!= PCI_ANY_ID
51 && asihpi_pci_tbl
[i
].subvendor
!=
52 pci_info
->subsys_vendor_id
)
54 if (asihpi_pci_tbl
[i
].subdevice
!= PCI_ANY_ID
55 && asihpi_pci_tbl
[i
].subdevice
!=
56 pci_info
->subsys_device_id
)
59 HPI_DEBUG_LOG(DEBUG
, " %x,%lu\n", i
,
60 asihpi_pci_tbl
[i
].driver_data
);
61 return (hpi_handler_func
*) asihpi_pci_tbl
[i
].driver_data
;
67 static inline void hw_entry_point(struct hpi_message
*phm
,
68 struct hpi_response
*phr
)
73 if (phm
->adapter_index
< HPI_MAX_ADAPTERS
) {
74 ep
= (hpi_handler_func
*) hpi_entry_points
[phm
->
77 HPI_DEBUG_MESSAGE(DEBUG
, phm
);
79 HPI_DEBUG_RESPONSE(phr
);
83 hpi_init_response(phr
, phm
->object
, phm
->function
,
84 HPI_ERROR_PROCESSING_MESSAGE
);
87 static void adapter_open(struct hpi_message
*phm
, struct hpi_response
*phr
);
88 static void adapter_close(struct hpi_message
*phm
, struct hpi_response
*phr
);
90 static void mixer_open(struct hpi_message
*phm
, struct hpi_response
*phr
);
91 static void mixer_close(struct hpi_message
*phm
, struct hpi_response
*phr
);
93 static void outstream_open(struct hpi_message
*phm
, struct hpi_response
*phr
,
95 static void outstream_close(struct hpi_message
*phm
, struct hpi_response
*phr
,
97 static void instream_open(struct hpi_message
*phm
, struct hpi_response
*phr
,
99 static void instream_close(struct hpi_message
*phm
, struct hpi_response
*phr
,
102 static void HPIMSGX__reset(u16 adapter_index
);
103 static u16
HPIMSGX__init(struct hpi_message
*phm
, struct hpi_response
*phr
);
104 static void HPIMSGX__cleanup(u16 adapter_index
, void *h_owner
);
106 #ifndef DISABLE_PRAGMA_PACK1
107 #pragma pack(push, 1)
110 struct hpi_subsys_response
{
111 struct hpi_response_header h
;
112 struct hpi_subsys_res s
;
115 struct hpi_adapter_response
{
116 struct hpi_response_header h
;
117 struct hpi_adapter_res a
;
120 struct hpi_mixer_response
{
121 struct hpi_response_header h
;
122 struct hpi_mixer_res m
;
125 struct hpi_stream_response
{
126 struct hpi_response_header h
;
127 struct hpi_stream_res d
;
130 struct adapter_info
{
136 struct asi_open_state
{
141 #ifndef DISABLE_PRAGMA_PACK1
146 static struct hpi_adapter_response rESP_HPI_ADAPTER_OPEN
[HPI_MAX_ADAPTERS
];
148 static struct hpi_stream_response
149 rESP_HPI_OSTREAM_OPEN
[HPI_MAX_ADAPTERS
][HPI_MAX_STREAMS
];
151 static struct hpi_stream_response
152 rESP_HPI_ISTREAM_OPEN
[HPI_MAX_ADAPTERS
][HPI_MAX_STREAMS
];
154 static struct hpi_mixer_response rESP_HPI_MIXER_OPEN
[HPI_MAX_ADAPTERS
];
156 static struct hpi_subsys_response gRESP_HPI_SUBSYS_FIND_ADAPTERS
;
158 static struct adapter_info aDAPTER_INFO
[HPI_MAX_ADAPTERS
];
160 /* use these to keep track of opens from user mode apps/DLLs */
161 static struct asi_open_state
162 outstream_user_open
[HPI_MAX_ADAPTERS
][HPI_MAX_STREAMS
];
164 static struct asi_open_state
165 instream_user_open
[HPI_MAX_ADAPTERS
][HPI_MAX_STREAMS
];
167 static void subsys_message(struct hpi_message
*phm
, struct hpi_response
*phr
,
170 switch (phm
->function
) {
171 case HPI_SUBSYS_GET_VERSION
:
172 hpi_init_response(phr
, HPI_OBJ_SUBSYSTEM
,
173 HPI_SUBSYS_GET_VERSION
, 0);
174 phr
->u
.s
.version
= HPI_VER
>> 8; /* return major.minor */
175 phr
->u
.s
.data
= HPI_VER
; /* return major.minor.release */
177 case HPI_SUBSYS_OPEN
:
178 /*do not propagate the message down the chain */
179 hpi_init_response(phr
, HPI_OBJ_SUBSYSTEM
, HPI_SUBSYS_OPEN
, 0);
181 case HPI_SUBSYS_CLOSE
:
182 /*do not propagate the message down the chain */
183 hpi_init_response(phr
, HPI_OBJ_SUBSYSTEM
, HPI_SUBSYS_CLOSE
,
185 HPIMSGX__cleanup(HPIMSGX_ALLADAPTERS
, h_owner
);
187 case HPI_SUBSYS_DRIVER_LOAD
:
188 /* Initialize this module's internal state */
189 hpios_msgxlock_init(&msgx_lock
);
190 memset(&hpi_entry_points
, 0, sizeof(hpi_entry_points
));
191 hpios_locked_mem_init();
192 /* Init subsys_findadapters response to no-adapters */
193 HPIMSGX__reset(HPIMSGX_ALLADAPTERS
);
194 hpi_init_response(phr
, HPI_OBJ_SUBSYSTEM
,
195 HPI_SUBSYS_DRIVER_LOAD
, 0);
196 /* individual HPIs dont implement driver load */
197 HPI_COMMON(phm
, phr
);
199 case HPI_SUBSYS_DRIVER_UNLOAD
:
200 HPI_COMMON(phm
, phr
);
201 HPIMSGX__cleanup(HPIMSGX_ALLADAPTERS
, h_owner
);
202 hpios_locked_mem_free_all();
203 hpi_init_response(phr
, HPI_OBJ_SUBSYSTEM
,
204 HPI_SUBSYS_DRIVER_UNLOAD
, 0);
207 case HPI_SUBSYS_GET_INFO
:
208 HPI_COMMON(phm
, phr
);
211 case HPI_SUBSYS_FIND_ADAPTERS
:
212 memcpy(phr
, &gRESP_HPI_SUBSYS_FIND_ADAPTERS
,
213 sizeof(gRESP_HPI_SUBSYS_FIND_ADAPTERS
));
215 case HPI_SUBSYS_GET_NUM_ADAPTERS
:
216 memcpy(phr
, &gRESP_HPI_SUBSYS_FIND_ADAPTERS
,
217 sizeof(gRESP_HPI_SUBSYS_FIND_ADAPTERS
));
218 phr
->function
= HPI_SUBSYS_GET_NUM_ADAPTERS
;
220 case HPI_SUBSYS_GET_ADAPTER
:
222 int count
= phm
->adapter_index
;
224 hpi_init_response(phr
, HPI_OBJ_SUBSYSTEM
,
225 HPI_SUBSYS_GET_ADAPTER
, 0);
227 /* This is complicated by the fact that we want to
228 * "skip" 0's in the adapter list.
229 * First, make sure we are pointing to a
230 * non-zero adapter type.
232 while (gRESP_HPI_SUBSYS_FIND_ADAPTERS
.
233 s
.aw_adapter_list
[index
] == 0) {
235 if (index
>= HPI_MAX_ADAPTERS
)
239 /* move on to the next adapter */
241 if (index
>= HPI_MAX_ADAPTERS
)
243 while (gRESP_HPI_SUBSYS_FIND_ADAPTERS
.
244 s
.aw_adapter_list
[index
] == 0) {
246 if (index
>= HPI_MAX_ADAPTERS
)
252 if (index
< HPI_MAX_ADAPTERS
) {
253 phr
->u
.s
.adapter_index
= (u16
)index
;
254 phr
->u
.s
.aw_adapter_list
[0] =
255 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.
256 s
.aw_adapter_list
[index
];
258 phr
->u
.s
.adapter_index
= 0;
259 phr
->u
.s
.aw_adapter_list
[0] = 0;
260 phr
->error
= HPI_ERROR_BAD_ADAPTER_NUMBER
;
264 case HPI_SUBSYS_CREATE_ADAPTER
:
265 HPIMSGX__init(phm
, phr
);
267 case HPI_SUBSYS_DELETE_ADAPTER
:
268 HPIMSGX__cleanup(phm
->adapter_index
, h_owner
);
270 struct hpi_message hm
;
271 struct hpi_response hr
;
272 /* call to HPI_ADAPTER_CLOSE */
273 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_ADAPTER
,
275 hm
.adapter_index
= phm
->adapter_index
;
276 hw_entry_point(&hm
, &hr
);
278 hw_entry_point(phm
, phr
);
279 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.
280 aw_adapter_list
[phm
->adapter_index
]
282 hpi_entry_points
[phm
->adapter_index
] = NULL
;
285 hw_entry_point(phm
, phr
);
290 static void adapter_message(struct hpi_message
*phm
, struct hpi_response
*phr
,
293 switch (phm
->function
) {
294 case HPI_ADAPTER_OPEN
:
295 adapter_open(phm
, phr
);
297 case HPI_ADAPTER_CLOSE
:
298 adapter_close(phm
, phr
);
301 hw_entry_point(phm
, phr
);
306 static void mixer_message(struct hpi_message
*phm
, struct hpi_response
*phr
)
308 switch (phm
->function
) {
310 mixer_open(phm
, phr
);
312 case HPI_MIXER_CLOSE
:
313 mixer_close(phm
, phr
);
316 hw_entry_point(phm
, phr
);
321 static void outstream_message(struct hpi_message
*phm
,
322 struct hpi_response
*phr
, void *h_owner
)
324 if (phm
->obj_index
>= aDAPTER_INFO
[phm
->adapter_index
].num_outstreams
) {
325 hpi_init_response(phr
, HPI_OBJ_OSTREAM
, phm
->function
,
326 HPI_ERROR_INVALID_OBJ_INDEX
);
330 switch (phm
->function
) {
331 case HPI_OSTREAM_OPEN
:
332 outstream_open(phm
, phr
, h_owner
);
334 case HPI_OSTREAM_CLOSE
:
335 outstream_close(phm
, phr
, h_owner
);
338 hw_entry_point(phm
, phr
);
343 static void instream_message(struct hpi_message
*phm
,
344 struct hpi_response
*phr
, void *h_owner
)
346 if (phm
->obj_index
>= aDAPTER_INFO
[phm
->adapter_index
].num_instreams
) {
347 hpi_init_response(phr
, HPI_OBJ_ISTREAM
, phm
->function
,
348 HPI_ERROR_INVALID_OBJ_INDEX
);
352 switch (phm
->function
) {
353 case HPI_ISTREAM_OPEN
:
354 instream_open(phm
, phr
, h_owner
);
356 case HPI_ISTREAM_CLOSE
:
357 instream_close(phm
, phr
, h_owner
);
360 hw_entry_point(phm
, phr
);
365 /* NOTE: HPI_Message() must be defined in the driver as a wrapper for
366 * HPI_MessageEx so that functions in hpifunc.c compile.
368 void hpi_send_recv_ex(struct hpi_message
*phm
, struct hpi_response
*phr
,
371 HPI_DEBUG_MESSAGE(DEBUG
, phm
);
373 if (phm
->type
!= HPI_TYPE_MESSAGE
) {
374 hpi_init_response(phr
, phm
->object
, phm
->function
,
375 HPI_ERROR_INVALID_TYPE
);
379 if (phm
->adapter_index
>= HPI_MAX_ADAPTERS
380 && phm
->adapter_index
!= HPIMSGX_ALLADAPTERS
) {
381 hpi_init_response(phr
, phm
->object
, phm
->function
,
382 HPI_ERROR_BAD_ADAPTER_NUMBER
);
386 switch (phm
->object
) {
387 case HPI_OBJ_SUBSYSTEM
:
388 subsys_message(phm
, phr
, h_owner
);
391 case HPI_OBJ_ADAPTER
:
392 adapter_message(phm
, phr
, h_owner
);
396 mixer_message(phm
, phr
);
399 case HPI_OBJ_OSTREAM
:
400 outstream_message(phm
, phr
, h_owner
);
403 case HPI_OBJ_ISTREAM
:
404 instream_message(phm
, phr
, h_owner
);
408 hw_entry_point(phm
, phr
);
411 HPI_DEBUG_RESPONSE(phr
);
413 if (phr
->error
>= HPI_ERROR_BACKEND_BASE
) {
417 HPI_DEBUG_MESSAGE(ERROR
, phm
);
419 if (phm
->adapter_index
< HPI_MAX_ADAPTERS
)
420 ep
= hpi_entry_points
[phm
->adapter_index
];
422 /* Don't need this? Have adapter index in debug info
423 Know at driver load time index->backend mapping */
425 ep_name
= "HPI_6000";
426 else if (ep
== HPI_6205
)
427 ep_name
= "HPI_6205";
431 HPI_DEBUG_LOG(ERROR
, "HPI %s response - error# %d\n", ep_name
,
434 if (hpi_debug_level
>= HPI_DEBUG_LEVEL_VERBOSE
)
435 hpi_debug_data((u16
*)phm
,
436 sizeof(*phm
) / sizeof(u16
));
441 static void adapter_open(struct hpi_message
*phm
, struct hpi_response
*phr
)
443 HPI_DEBUG_LOG(VERBOSE
, "adapter_open\n");
444 memcpy(phr
, &rESP_HPI_ADAPTER_OPEN
[phm
->adapter_index
],
445 sizeof(rESP_HPI_ADAPTER_OPEN
[0]));
448 static void adapter_close(struct hpi_message
*phm
, struct hpi_response
*phr
)
450 HPI_DEBUG_LOG(VERBOSE
, "adapter_close\n");
451 hpi_init_response(phr
, HPI_OBJ_ADAPTER
, HPI_ADAPTER_CLOSE
, 0);
454 static void mixer_open(struct hpi_message
*phm
, struct hpi_response
*phr
)
456 memcpy(phr
, &rESP_HPI_MIXER_OPEN
[phm
->adapter_index
],
457 sizeof(rESP_HPI_MIXER_OPEN
[0]));
460 static void mixer_close(struct hpi_message
*phm
, struct hpi_response
*phr
)
462 hpi_init_response(phr
, HPI_OBJ_MIXER
, HPI_MIXER_CLOSE
, 0);
465 static void instream_open(struct hpi_message
*phm
, struct hpi_response
*phr
,
469 struct hpi_message hm
;
470 struct hpi_response hr
;
472 hpi_init_response(phr
, HPI_OBJ_ISTREAM
, HPI_ISTREAM_OPEN
, 0);
474 hpios_msgxlock_lock(&msgx_lock
);
476 if (instream_user_open
[phm
->adapter_index
][phm
->obj_index
].open_flag
)
477 phr
->error
= HPI_ERROR_OBJ_ALREADY_OPEN
;
478 else if (rESP_HPI_ISTREAM_OPEN
[phm
->adapter_index
]
479 [phm
->obj_index
].h
.error
)
481 &rESP_HPI_ISTREAM_OPEN
[phm
->adapter_index
][phm
->
483 sizeof(rESP_HPI_ISTREAM_OPEN
[0][0]));
485 instream_user_open
[phm
->adapter_index
][phm
->
486 obj_index
].open_flag
= 1;
487 hpios_msgxlock_un_lock(&msgx_lock
);
490 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_ISTREAM
,
492 hm
.adapter_index
= phm
->adapter_index
;
493 hm
.obj_index
= phm
->obj_index
;
494 hw_entry_point(&hm
, &hr
);
496 hpios_msgxlock_lock(&msgx_lock
);
498 instream_user_open
[phm
->adapter_index
][phm
->
499 obj_index
].open_flag
= 0;
500 phr
->error
= hr
.error
;
502 instream_user_open
[phm
->adapter_index
][phm
->
503 obj_index
].open_flag
= 1;
504 instream_user_open
[phm
->adapter_index
][phm
->
505 obj_index
].h_owner
= h_owner
;
507 &rESP_HPI_ISTREAM_OPEN
[phm
->adapter_index
]
509 sizeof(rESP_HPI_ISTREAM_OPEN
[0][0]));
512 hpios_msgxlock_un_lock(&msgx_lock
);
515 static void instream_close(struct hpi_message
*phm
, struct hpi_response
*phr
,
519 struct hpi_message hm
;
520 struct hpi_response hr
;
522 hpi_init_response(phr
, HPI_OBJ_ISTREAM
, HPI_ISTREAM_CLOSE
, 0);
524 hpios_msgxlock_lock(&msgx_lock
);
526 instream_user_open
[phm
->adapter_index
][phm
->
527 obj_index
].h_owner
) {
528 /* HPI_DEBUG_LOG(INFO,"closing adapter %d "
529 "instream %d owned by %p\n",
530 phm->wAdapterIndex, phm->wObjIndex, hOwner); */
531 instream_user_open
[phm
->adapter_index
][phm
->
532 obj_index
].h_owner
= NULL
;
533 hpios_msgxlock_un_lock(&msgx_lock
);
535 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_ISTREAM
,
537 hm
.adapter_index
= phm
->adapter_index
;
538 hm
.obj_index
= phm
->obj_index
;
539 hw_entry_point(&hm
, &hr
);
540 hpios_msgxlock_lock(&msgx_lock
);
542 instream_user_open
[phm
->adapter_index
][phm
->
543 obj_index
].h_owner
= h_owner
;
544 phr
->error
= hr
.error
;
546 instream_user_open
[phm
->adapter_index
][phm
->
547 obj_index
].open_flag
= 0;
548 instream_user_open
[phm
->adapter_index
][phm
->
549 obj_index
].h_owner
= NULL
;
552 HPI_DEBUG_LOG(WARNING
,
553 "%p trying to close %d instream %d owned by %p\n",
554 h_owner
, phm
->adapter_index
, phm
->obj_index
,
555 instream_user_open
[phm
->adapter_index
][phm
->
557 phr
->error
= HPI_ERROR_OBJ_NOT_OPEN
;
559 hpios_msgxlock_un_lock(&msgx_lock
);
562 static void outstream_open(struct hpi_message
*phm
, struct hpi_response
*phr
,
566 struct hpi_message hm
;
567 struct hpi_response hr
;
569 hpi_init_response(phr
, HPI_OBJ_OSTREAM
, HPI_OSTREAM_OPEN
, 0);
571 hpios_msgxlock_lock(&msgx_lock
);
573 if (outstream_user_open
[phm
->adapter_index
][phm
->obj_index
].open_flag
)
574 phr
->error
= HPI_ERROR_OBJ_ALREADY_OPEN
;
575 else if (rESP_HPI_OSTREAM_OPEN
[phm
->adapter_index
]
576 [phm
->obj_index
].h
.error
)
578 &rESP_HPI_OSTREAM_OPEN
[phm
->adapter_index
][phm
->
580 sizeof(rESP_HPI_OSTREAM_OPEN
[0][0]));
582 outstream_user_open
[phm
->adapter_index
][phm
->
583 obj_index
].open_flag
= 1;
584 hpios_msgxlock_un_lock(&msgx_lock
);
587 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_OSTREAM
,
589 hm
.adapter_index
= phm
->adapter_index
;
590 hm
.obj_index
= phm
->obj_index
;
591 hw_entry_point(&hm
, &hr
);
593 hpios_msgxlock_lock(&msgx_lock
);
595 outstream_user_open
[phm
->adapter_index
][phm
->
596 obj_index
].open_flag
= 0;
597 phr
->error
= hr
.error
;
599 outstream_user_open
[phm
->adapter_index
][phm
->
600 obj_index
].open_flag
= 1;
601 outstream_user_open
[phm
->adapter_index
][phm
->
602 obj_index
].h_owner
= h_owner
;
604 &rESP_HPI_OSTREAM_OPEN
[phm
->adapter_index
]
606 sizeof(rESP_HPI_OSTREAM_OPEN
[0][0]));
609 hpios_msgxlock_un_lock(&msgx_lock
);
612 static void outstream_close(struct hpi_message
*phm
, struct hpi_response
*phr
,
616 struct hpi_message hm
;
617 struct hpi_response hr
;
619 hpi_init_response(phr
, HPI_OBJ_OSTREAM
, HPI_OSTREAM_CLOSE
, 0);
621 hpios_msgxlock_lock(&msgx_lock
);
624 outstream_user_open
[phm
->adapter_index
][phm
->
625 obj_index
].h_owner
) {
626 /* HPI_DEBUG_LOG(INFO,"closing adapter %d "
627 "outstream %d owned by %p\n",
628 phm->wAdapterIndex, phm->wObjIndex, hOwner); */
629 outstream_user_open
[phm
->adapter_index
][phm
->
630 obj_index
].h_owner
= NULL
;
631 hpios_msgxlock_un_lock(&msgx_lock
);
633 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_OSTREAM
,
635 hm
.adapter_index
= phm
->adapter_index
;
636 hm
.obj_index
= phm
->obj_index
;
637 hw_entry_point(&hm
, &hr
);
638 hpios_msgxlock_lock(&msgx_lock
);
640 outstream_user_open
[phm
->adapter_index
][phm
->
641 obj_index
].h_owner
= h_owner
;
642 phr
->error
= hr
.error
;
644 outstream_user_open
[phm
->adapter_index
][phm
->
645 obj_index
].open_flag
= 0;
646 outstream_user_open
[phm
->adapter_index
][phm
->
647 obj_index
].h_owner
= NULL
;
650 HPI_DEBUG_LOG(WARNING
,
651 "%p trying to close %d outstream %d owned by %p\n",
652 h_owner
, phm
->adapter_index
, phm
->obj_index
,
653 outstream_user_open
[phm
->adapter_index
][phm
->
655 phr
->error
= HPI_ERROR_OBJ_NOT_OPEN
;
657 hpios_msgxlock_un_lock(&msgx_lock
);
660 static u16
adapter_prepare(u16 adapter
)
662 struct hpi_message hm
;
663 struct hpi_response hr
;
665 /* Open the adapter and streams */
668 /* call to HPI_ADAPTER_OPEN */
669 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_ADAPTER
,
671 hm
.adapter_index
= adapter
;
672 hw_entry_point(&hm
, &hr
);
673 memcpy(&rESP_HPI_ADAPTER_OPEN
[adapter
], &hr
,
674 sizeof(rESP_HPI_ADAPTER_OPEN
[0]));
678 /* call to HPI_ADAPTER_GET_INFO */
679 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_ADAPTER
,
680 HPI_ADAPTER_GET_INFO
);
681 hm
.adapter_index
= adapter
;
682 hw_entry_point(&hm
, &hr
);
686 aDAPTER_INFO
[adapter
].num_outstreams
= hr
.u
.a
.num_outstreams
;
687 aDAPTER_INFO
[adapter
].num_instreams
= hr
.u
.a
.num_instreams
;
688 aDAPTER_INFO
[adapter
].type
= hr
.u
.a
.adapter_type
;
690 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.aw_adapter_list
[adapter
] =
692 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.num_adapters
++;
693 if (gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.num_adapters
> HPI_MAX_ADAPTERS
)
694 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.num_adapters
=
697 /* call to HPI_OSTREAM_OPEN */
698 for (i
= 0; i
< aDAPTER_INFO
[adapter
].num_outstreams
; i
++) {
699 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_OSTREAM
,
701 hm
.adapter_index
= adapter
;
703 hw_entry_point(&hm
, &hr
);
704 memcpy(&rESP_HPI_OSTREAM_OPEN
[adapter
][i
], &hr
,
705 sizeof(rESP_HPI_OSTREAM_OPEN
[0][0]));
706 outstream_user_open
[adapter
][i
].open_flag
= 0;
707 outstream_user_open
[adapter
][i
].h_owner
= NULL
;
710 /* call to HPI_ISTREAM_OPEN */
711 for (i
= 0; i
< aDAPTER_INFO
[adapter
].num_instreams
; i
++) {
712 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_ISTREAM
,
714 hm
.adapter_index
= adapter
;
716 hw_entry_point(&hm
, &hr
);
717 memcpy(&rESP_HPI_ISTREAM_OPEN
[adapter
][i
], &hr
,
718 sizeof(rESP_HPI_ISTREAM_OPEN
[0][0]));
719 instream_user_open
[adapter
][i
].open_flag
= 0;
720 instream_user_open
[adapter
][i
].h_owner
= NULL
;
723 /* call to HPI_MIXER_OPEN */
724 hpi_init_message_response(&hm
, &hr
, HPI_OBJ_MIXER
, HPI_MIXER_OPEN
);
725 hm
.adapter_index
= adapter
;
726 hw_entry_point(&hm
, &hr
);
727 memcpy(&rESP_HPI_MIXER_OPEN
[adapter
], &hr
,
728 sizeof(rESP_HPI_MIXER_OPEN
[0]));
730 return gRESP_HPI_SUBSYS_FIND_ADAPTERS
.h
.error
;
733 static void HPIMSGX__reset(u16 adapter_index
)
737 struct hpi_response hr
;
739 if (adapter_index
== HPIMSGX_ALLADAPTERS
) {
740 /* reset all responses to contain errors */
741 hpi_init_response(&hr
, HPI_OBJ_SUBSYSTEM
,
742 HPI_SUBSYS_FIND_ADAPTERS
, 0);
743 memcpy(&gRESP_HPI_SUBSYS_FIND_ADAPTERS
, &hr
,
744 sizeof(&gRESP_HPI_SUBSYS_FIND_ADAPTERS
));
746 for (adapter
= 0; adapter
< HPI_MAX_ADAPTERS
; adapter
++) {
748 hpi_init_response(&hr
, HPI_OBJ_ADAPTER
,
749 HPI_ADAPTER_OPEN
, HPI_ERROR_BAD_ADAPTER
);
750 memcpy(&rESP_HPI_ADAPTER_OPEN
[adapter
], &hr
,
751 sizeof(rESP_HPI_ADAPTER_OPEN
[adapter
]));
753 hpi_init_response(&hr
, HPI_OBJ_MIXER
, HPI_MIXER_OPEN
,
754 HPI_ERROR_INVALID_OBJ
);
755 memcpy(&rESP_HPI_MIXER_OPEN
[adapter
], &hr
,
756 sizeof(rESP_HPI_MIXER_OPEN
[adapter
]));
758 for (i
= 0; i
< HPI_MAX_STREAMS
; i
++) {
759 hpi_init_response(&hr
, HPI_OBJ_OSTREAM
,
761 HPI_ERROR_INVALID_OBJ
);
762 memcpy(&rESP_HPI_OSTREAM_OPEN
[adapter
][i
],
764 sizeof(rESP_HPI_OSTREAM_OPEN
[adapter
]
766 hpi_init_response(&hr
, HPI_OBJ_ISTREAM
,
768 HPI_ERROR_INVALID_OBJ
);
769 memcpy(&rESP_HPI_ISTREAM_OPEN
[adapter
][i
],
771 sizeof(rESP_HPI_ISTREAM_OPEN
[adapter
]
775 } else if (adapter_index
< HPI_MAX_ADAPTERS
) {
776 rESP_HPI_ADAPTER_OPEN
[adapter_index
].h
.error
=
777 HPI_ERROR_BAD_ADAPTER
;
778 rESP_HPI_MIXER_OPEN
[adapter_index
].h
.error
=
779 HPI_ERROR_INVALID_OBJ
;
780 for (i
= 0; i
< HPI_MAX_STREAMS
; i
++) {
781 rESP_HPI_OSTREAM_OPEN
[adapter_index
][i
].h
.error
=
782 HPI_ERROR_INVALID_OBJ
;
783 rESP_HPI_ISTREAM_OPEN
[adapter_index
][i
].h
.error
=
784 HPI_ERROR_INVALID_OBJ
;
786 if (gRESP_HPI_SUBSYS_FIND_ADAPTERS
.
787 s
.aw_adapter_list
[adapter_index
]) {
788 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.
789 s
.aw_adapter_list
[adapter_index
] = 0;
790 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.num_adapters
--;
795 static u16
HPIMSGX__init(struct hpi_message
*phm
,
796 /* HPI_SUBSYS_CREATE_ADAPTER structure with */
797 /* resource list or NULL=find all */
798 struct hpi_response
*phr
799 /* response from HPI_ADAPTER_GET_INFO */
802 hpi_handler_func
*entry_point_func
;
803 struct hpi_response hr
;
805 if (gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.num_adapters
>= HPI_MAX_ADAPTERS
)
806 return HPI_ERROR_BAD_ADAPTER_NUMBER
;
808 /* Init response here so we can pass in previous adapter list */
809 hpi_init_response(&hr
, phm
->object
, phm
->function
,
810 HPI_ERROR_INVALID_OBJ
);
811 memcpy(hr
.u
.s
.aw_adapter_list
,
812 gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.aw_adapter_list
,
813 sizeof(gRESP_HPI_SUBSYS_FIND_ADAPTERS
.s
.aw_adapter_list
));
816 hpi_lookup_entry_point_function(phm
->u
.s
.resource
.r
.pci
);
818 if (entry_point_func
) {
819 HPI_DEBUG_MESSAGE(DEBUG
, phm
);
820 entry_point_func(phm
, &hr
);
822 phr
->error
= HPI_ERROR_PROCESSING_MESSAGE
;
826 /* the adapter was created succesfully
827 save the mapping for future use */
828 hpi_entry_points
[hr
.u
.s
.adapter_index
] = entry_point_func
;
829 /* prepare adapter (pre-open streams etc.) */
831 "HPI_SUBSYS_CREATE_ADAPTER successful,"
832 " preparing adapter\n");
833 adapter_prepare(hr
.u
.s
.adapter_index
);
835 memcpy(phr
, &hr
, hr
.size
);
839 static void HPIMSGX__cleanup(u16 adapter_index
, void *h_owner
)
841 int i
, adapter
, adapter_limit
;
846 if (adapter_index
== HPIMSGX_ALLADAPTERS
) {
848 adapter_limit
= HPI_MAX_ADAPTERS
;
850 adapter
= adapter_index
;
851 adapter_limit
= adapter
+ 1;
854 for (; adapter
< adapter_limit
; adapter
++) {
855 /* printk(KERN_INFO "Cleanup adapter #%d\n",wAdapter); */
856 for (i
= 0; i
< HPI_MAX_STREAMS
; i
++) {
858 outstream_user_open
[adapter
][i
].h_owner
) {
859 struct hpi_message hm
;
860 struct hpi_response hr
;
863 "close adapter %d ostream %d\n",
866 hpi_init_message_response(&hm
, &hr
,
867 HPI_OBJ_OSTREAM
, HPI_OSTREAM_RESET
);
868 hm
.adapter_index
= (u16
)adapter
;
869 hm
.obj_index
= (u16
)i
;
870 hw_entry_point(&hm
, &hr
);
872 hm
.function
= HPI_OSTREAM_HOSTBUFFER_FREE
;
873 hw_entry_point(&hm
, &hr
);
875 hm
.function
= HPI_OSTREAM_GROUP_RESET
;
876 hw_entry_point(&hm
, &hr
);
878 outstream_user_open
[adapter
][i
].open_flag
= 0;
879 outstream_user_open
[adapter
][i
].h_owner
=
882 if (h_owner
== instream_user_open
[adapter
][i
].h_owner
) {
883 struct hpi_message hm
;
884 struct hpi_response hr
;
887 "close adapter %d istream %d\n",
890 hpi_init_message_response(&hm
, &hr
,
891 HPI_OBJ_ISTREAM
, HPI_ISTREAM_RESET
);
892 hm
.adapter_index
= (u16
)adapter
;
893 hm
.obj_index
= (u16
)i
;
894 hw_entry_point(&hm
, &hr
);
896 hm
.function
= HPI_ISTREAM_HOSTBUFFER_FREE
;
897 hw_entry_point(&hm
, &hr
);
899 hm
.function
= HPI_ISTREAM_GROUP_RESET
;
900 hw_entry_point(&hm
, &hr
);
902 instream_user_open
[adapter
][i
].open_flag
= 0;
903 instream_user_open
[adapter
][i
].h_owner
= NULL
;