18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Driver for Digigram miXart soundcards 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * DSP firmware management 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Copyright (c) 2003 by Digigram <alsa@digigram.com> 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 118c2ecf20Sopenharmony_ci#include <linux/pci.h> 128c2ecf20Sopenharmony_ci#include <linux/firmware.h> 138c2ecf20Sopenharmony_ci#include <linux/vmalloc.h> 148c2ecf20Sopenharmony_ci#include <linux/slab.h> 158c2ecf20Sopenharmony_ci#include <linux/module.h> 168c2ecf20Sopenharmony_ci#include <linux/io.h> 178c2ecf20Sopenharmony_ci#include <sound/core.h> 188c2ecf20Sopenharmony_ci#include "mixart.h" 198c2ecf20Sopenharmony_ci#include "mixart_mixer.h" 208c2ecf20Sopenharmony_ci#include "mixart_core.h" 218c2ecf20Sopenharmony_ci#include "mixart_hwdep.h" 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_ci/** 258c2ecf20Sopenharmony_ci * wait for a value on a peudo register, exit with a timeout 268c2ecf20Sopenharmony_ci * 278c2ecf20Sopenharmony_ci * @mgr: pointer to miXart manager structure 288c2ecf20Sopenharmony_ci * @offset: unsigned pseudo_register base + offset of value 298c2ecf20Sopenharmony_ci * @is_egal: wait for the equal value 308c2ecf20Sopenharmony_ci * @value: value 318c2ecf20Sopenharmony_ci * @timeout: timeout in centisenconds 328c2ecf20Sopenharmony_ci */ 338c2ecf20Sopenharmony_cistatic int mixart_wait_nice_for_register_value(struct mixart_mgr *mgr, 348c2ecf20Sopenharmony_ci u32 offset, int is_egal, 358c2ecf20Sopenharmony_ci u32 value, unsigned long timeout) 368c2ecf20Sopenharmony_ci{ 378c2ecf20Sopenharmony_ci unsigned long end_time = jiffies + (timeout * HZ / 100); 388c2ecf20Sopenharmony_ci u32 read; 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_ci do { /* we may take too long time in this loop. 418c2ecf20Sopenharmony_ci * so give controls back to kernel if needed. 428c2ecf20Sopenharmony_ci */ 438c2ecf20Sopenharmony_ci cond_resched(); 448c2ecf20Sopenharmony_ci 458c2ecf20Sopenharmony_ci read = readl_be( MIXART_MEM( mgr, offset )); 468c2ecf20Sopenharmony_ci if(is_egal) { 478c2ecf20Sopenharmony_ci if(read == value) return 0; 488c2ecf20Sopenharmony_ci } 498c2ecf20Sopenharmony_ci else { /* wait for different value */ 508c2ecf20Sopenharmony_ci if(read != value) return 0; 518c2ecf20Sopenharmony_ci } 528c2ecf20Sopenharmony_ci } while ( time_after_eq(end_time, jiffies) ); 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_ci return -EBUSY; 558c2ecf20Sopenharmony_ci} 568c2ecf20Sopenharmony_ci 578c2ecf20Sopenharmony_ci 588c2ecf20Sopenharmony_ci/* 598c2ecf20Sopenharmony_ci structures needed to upload elf code packets 608c2ecf20Sopenharmony_ci */ 618c2ecf20Sopenharmony_cistruct snd_mixart_elf32_ehdr { 628c2ecf20Sopenharmony_ci u8 e_ident[16]; 638c2ecf20Sopenharmony_ci __be16 e_type; 648c2ecf20Sopenharmony_ci __be16 e_machine; 658c2ecf20Sopenharmony_ci __be32 e_version; 668c2ecf20Sopenharmony_ci __be32 e_entry; 678c2ecf20Sopenharmony_ci __be32 e_phoff; 688c2ecf20Sopenharmony_ci __be32 e_shoff; 698c2ecf20Sopenharmony_ci __be32 e_flags; 708c2ecf20Sopenharmony_ci __be16 e_ehsize; 718c2ecf20Sopenharmony_ci __be16 e_phentsize; 728c2ecf20Sopenharmony_ci __be16 e_phnum; 738c2ecf20Sopenharmony_ci __be16 e_shentsize; 748c2ecf20Sopenharmony_ci __be16 e_shnum; 758c2ecf20Sopenharmony_ci __be16 e_shstrndx; 768c2ecf20Sopenharmony_ci}; 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_cistruct snd_mixart_elf32_phdr { 798c2ecf20Sopenharmony_ci __be32 p_type; 808c2ecf20Sopenharmony_ci __be32 p_offset; 818c2ecf20Sopenharmony_ci __be32 p_vaddr; 828c2ecf20Sopenharmony_ci __be32 p_paddr; 838c2ecf20Sopenharmony_ci __be32 p_filesz; 848c2ecf20Sopenharmony_ci __be32 p_memsz; 858c2ecf20Sopenharmony_ci __be32 p_flags; 868c2ecf20Sopenharmony_ci __be32 p_align; 878c2ecf20Sopenharmony_ci}; 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_cistatic int mixart_load_elf(struct mixart_mgr *mgr, const struct firmware *dsp ) 908c2ecf20Sopenharmony_ci{ 918c2ecf20Sopenharmony_ci char elf32_magic_number[4] = {0x7f,'E','L','F'}; 928c2ecf20Sopenharmony_ci struct snd_mixart_elf32_ehdr *elf_header; 938c2ecf20Sopenharmony_ci int i; 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_ci elf_header = (struct snd_mixart_elf32_ehdr *)dsp->data; 968c2ecf20Sopenharmony_ci for( i=0; i<4; i++ ) 978c2ecf20Sopenharmony_ci if ( elf32_magic_number[i] != elf_header->e_ident[i] ) 988c2ecf20Sopenharmony_ci return -EINVAL; 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_ci if( elf_header->e_phoff != 0 ) { 1018c2ecf20Sopenharmony_ci struct snd_mixart_elf32_phdr elf_programheader; 1028c2ecf20Sopenharmony_ci 1038c2ecf20Sopenharmony_ci for( i=0; i < be16_to_cpu(elf_header->e_phnum); i++ ) { 1048c2ecf20Sopenharmony_ci u32 pos = be32_to_cpu(elf_header->e_phoff) + (u32)(i * be16_to_cpu(elf_header->e_phentsize)); 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci memcpy( &elf_programheader, dsp->data + pos, sizeof(elf_programheader) ); 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci if(elf_programheader.p_type != 0) { 1098c2ecf20Sopenharmony_ci if( elf_programheader.p_filesz != 0 ) { 1108c2ecf20Sopenharmony_ci memcpy_toio( MIXART_MEM( mgr, be32_to_cpu(elf_programheader.p_vaddr)), 1118c2ecf20Sopenharmony_ci dsp->data + be32_to_cpu( elf_programheader.p_offset ), 1128c2ecf20Sopenharmony_ci be32_to_cpu( elf_programheader.p_filesz )); 1138c2ecf20Sopenharmony_ci } 1148c2ecf20Sopenharmony_ci } 1158c2ecf20Sopenharmony_ci } 1168c2ecf20Sopenharmony_ci } 1178c2ecf20Sopenharmony_ci return 0; 1188c2ecf20Sopenharmony_ci} 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci/* 1218c2ecf20Sopenharmony_ci * get basic information and init miXart 1228c2ecf20Sopenharmony_ci */ 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_ci/* audio IDs for request to the board */ 1258c2ecf20Sopenharmony_ci#define MIXART_FIRST_ANA_AUDIO_ID 0 1268c2ecf20Sopenharmony_ci#define MIXART_FIRST_DIG_AUDIO_ID 8 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_cistatic int mixart_enum_connectors(struct mixart_mgr *mgr) 1298c2ecf20Sopenharmony_ci{ 1308c2ecf20Sopenharmony_ci u32 k; 1318c2ecf20Sopenharmony_ci int err; 1328c2ecf20Sopenharmony_ci struct mixart_msg request; 1338c2ecf20Sopenharmony_ci struct mixart_enum_connector_resp *connector; 1348c2ecf20Sopenharmony_ci struct mixart_audio_info_req *audio_info_req; 1358c2ecf20Sopenharmony_ci struct mixart_audio_info_resp *audio_info; 1368c2ecf20Sopenharmony_ci 1378c2ecf20Sopenharmony_ci connector = kmalloc(sizeof(*connector), GFP_KERNEL); 1388c2ecf20Sopenharmony_ci audio_info_req = kmalloc(sizeof(*audio_info_req), GFP_KERNEL); 1398c2ecf20Sopenharmony_ci audio_info = kmalloc(sizeof(*audio_info), GFP_KERNEL); 1408c2ecf20Sopenharmony_ci if (! connector || ! audio_info_req || ! audio_info) { 1418c2ecf20Sopenharmony_ci err = -ENOMEM; 1428c2ecf20Sopenharmony_ci goto __error; 1438c2ecf20Sopenharmony_ci } 1448c2ecf20Sopenharmony_ci 1458c2ecf20Sopenharmony_ci audio_info_req->line_max_level = MIXART_FLOAT_P_22_0_TO_HEX; 1468c2ecf20Sopenharmony_ci audio_info_req->micro_max_level = MIXART_FLOAT_M_20_0_TO_HEX; 1478c2ecf20Sopenharmony_ci audio_info_req->cd_max_level = MIXART_FLOAT____0_0_TO_HEX; 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_ci request.message_id = MSG_SYSTEM_ENUM_PLAY_CONNECTOR; 1508c2ecf20Sopenharmony_ci request.uid = (struct mixart_uid){0,0}; /* board num = 0 */ 1518c2ecf20Sopenharmony_ci request.data = NULL; 1528c2ecf20Sopenharmony_ci request.size = 0; 1538c2ecf20Sopenharmony_ci 1548c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(*connector), connector); 1558c2ecf20Sopenharmony_ci if((err < 0) || (connector->error_code) || (connector->uid_count > MIXART_MAX_PHYS_CONNECTORS)) { 1568c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 1578c2ecf20Sopenharmony_ci "error MSG_SYSTEM_ENUM_PLAY_CONNECTOR\n"); 1588c2ecf20Sopenharmony_ci err = -EINVAL; 1598c2ecf20Sopenharmony_ci goto __error; 1608c2ecf20Sopenharmony_ci } 1618c2ecf20Sopenharmony_ci 1628c2ecf20Sopenharmony_ci for(k=0; k < connector->uid_count; k++) { 1638c2ecf20Sopenharmony_ci struct mixart_pipe *pipe; 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_ci if(k < MIXART_FIRST_DIG_AUDIO_ID) { 1668c2ecf20Sopenharmony_ci pipe = &mgr->chip[k/2]->pipe_out_ana; 1678c2ecf20Sopenharmony_ci } else { 1688c2ecf20Sopenharmony_ci pipe = &mgr->chip[(k-MIXART_FIRST_DIG_AUDIO_ID)/2]->pipe_out_dig; 1698c2ecf20Sopenharmony_ci } 1708c2ecf20Sopenharmony_ci if(k & 1) { 1718c2ecf20Sopenharmony_ci pipe->uid_right_connector = connector->uid[k]; /* odd */ 1728c2ecf20Sopenharmony_ci } else { 1738c2ecf20Sopenharmony_ci pipe->uid_left_connector = connector->uid[k]; /* even */ 1748c2ecf20Sopenharmony_ci } 1758c2ecf20Sopenharmony_ci 1768c2ecf20Sopenharmony_ci /* dev_dbg(&mgr->pci->dev, "playback connector[%d].object_id = %x\n", k, connector->uid[k].object_id); */ 1778c2ecf20Sopenharmony_ci 1788c2ecf20Sopenharmony_ci /* TODO: really need send_msg MSG_CONNECTOR_GET_AUDIO_INFO for each connector ? perhaps for analog level caps ? */ 1798c2ecf20Sopenharmony_ci request.message_id = MSG_CONNECTOR_GET_AUDIO_INFO; 1808c2ecf20Sopenharmony_ci request.uid = connector->uid[k]; 1818c2ecf20Sopenharmony_ci request.data = audio_info_req; 1828c2ecf20Sopenharmony_ci request.size = sizeof(*audio_info_req); 1838c2ecf20Sopenharmony_ci 1848c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(*audio_info), audio_info); 1858c2ecf20Sopenharmony_ci if( err < 0 ) { 1868c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 1878c2ecf20Sopenharmony_ci "error MSG_CONNECTOR_GET_AUDIO_INFO\n"); 1888c2ecf20Sopenharmony_ci goto __error; 1898c2ecf20Sopenharmony_ci } 1908c2ecf20Sopenharmony_ci /*dev_dbg(&mgr->pci->dev, "play analog_info.analog_level_present = %x\n", audio_info->info.analog_info.analog_level_present);*/ 1918c2ecf20Sopenharmony_ci } 1928c2ecf20Sopenharmony_ci 1938c2ecf20Sopenharmony_ci request.message_id = MSG_SYSTEM_ENUM_RECORD_CONNECTOR; 1948c2ecf20Sopenharmony_ci request.uid = (struct mixart_uid){0,0}; /* board num = 0 */ 1958c2ecf20Sopenharmony_ci request.data = NULL; 1968c2ecf20Sopenharmony_ci request.size = 0; 1978c2ecf20Sopenharmony_ci 1988c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(*connector), connector); 1998c2ecf20Sopenharmony_ci if((err < 0) || (connector->error_code) || (connector->uid_count > MIXART_MAX_PHYS_CONNECTORS)) { 2008c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 2018c2ecf20Sopenharmony_ci "error MSG_SYSTEM_ENUM_RECORD_CONNECTOR\n"); 2028c2ecf20Sopenharmony_ci err = -EINVAL; 2038c2ecf20Sopenharmony_ci goto __error; 2048c2ecf20Sopenharmony_ci } 2058c2ecf20Sopenharmony_ci 2068c2ecf20Sopenharmony_ci for(k=0; k < connector->uid_count; k++) { 2078c2ecf20Sopenharmony_ci struct mixart_pipe *pipe; 2088c2ecf20Sopenharmony_ci 2098c2ecf20Sopenharmony_ci if(k < MIXART_FIRST_DIG_AUDIO_ID) { 2108c2ecf20Sopenharmony_ci pipe = &mgr->chip[k/2]->pipe_in_ana; 2118c2ecf20Sopenharmony_ci } else { 2128c2ecf20Sopenharmony_ci pipe = &mgr->chip[(k-MIXART_FIRST_DIG_AUDIO_ID)/2]->pipe_in_dig; 2138c2ecf20Sopenharmony_ci } 2148c2ecf20Sopenharmony_ci if(k & 1) { 2158c2ecf20Sopenharmony_ci pipe->uid_right_connector = connector->uid[k]; /* odd */ 2168c2ecf20Sopenharmony_ci } else { 2178c2ecf20Sopenharmony_ci pipe->uid_left_connector = connector->uid[k]; /* even */ 2188c2ecf20Sopenharmony_ci } 2198c2ecf20Sopenharmony_ci 2208c2ecf20Sopenharmony_ci /* dev_dbg(&mgr->pci->dev, "capture connector[%d].object_id = %x\n", k, connector->uid[k].object_id); */ 2218c2ecf20Sopenharmony_ci 2228c2ecf20Sopenharmony_ci /* TODO: really need send_msg MSG_CONNECTOR_GET_AUDIO_INFO for each connector ? perhaps for analog level caps ? */ 2238c2ecf20Sopenharmony_ci request.message_id = MSG_CONNECTOR_GET_AUDIO_INFO; 2248c2ecf20Sopenharmony_ci request.uid = connector->uid[k]; 2258c2ecf20Sopenharmony_ci request.data = audio_info_req; 2268c2ecf20Sopenharmony_ci request.size = sizeof(*audio_info_req); 2278c2ecf20Sopenharmony_ci 2288c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(*audio_info), audio_info); 2298c2ecf20Sopenharmony_ci if( err < 0 ) { 2308c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 2318c2ecf20Sopenharmony_ci "error MSG_CONNECTOR_GET_AUDIO_INFO\n"); 2328c2ecf20Sopenharmony_ci goto __error; 2338c2ecf20Sopenharmony_ci } 2348c2ecf20Sopenharmony_ci /*dev_dbg(&mgr->pci->dev, "rec analog_info.analog_level_present = %x\n", audio_info->info.analog_info.analog_level_present);*/ 2358c2ecf20Sopenharmony_ci } 2368c2ecf20Sopenharmony_ci err = 0; 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci __error: 2398c2ecf20Sopenharmony_ci kfree(connector); 2408c2ecf20Sopenharmony_ci kfree(audio_info_req); 2418c2ecf20Sopenharmony_ci kfree(audio_info); 2428c2ecf20Sopenharmony_ci 2438c2ecf20Sopenharmony_ci return err; 2448c2ecf20Sopenharmony_ci} 2458c2ecf20Sopenharmony_ci 2468c2ecf20Sopenharmony_cistatic int mixart_enum_physio(struct mixart_mgr *mgr) 2478c2ecf20Sopenharmony_ci{ 2488c2ecf20Sopenharmony_ci u32 k; 2498c2ecf20Sopenharmony_ci int err; 2508c2ecf20Sopenharmony_ci struct mixart_msg request; 2518c2ecf20Sopenharmony_ci struct mixart_uid get_console_mgr; 2528c2ecf20Sopenharmony_ci struct mixart_return_uid console_mgr; 2538c2ecf20Sopenharmony_ci struct mixart_uid_enumeration phys_io; 2548c2ecf20Sopenharmony_ci 2558c2ecf20Sopenharmony_ci /* get the uid for the console manager */ 2568c2ecf20Sopenharmony_ci get_console_mgr.object_id = 0; 2578c2ecf20Sopenharmony_ci get_console_mgr.desc = MSG_CONSOLE_MANAGER | 0; /* cardindex = 0 */ 2588c2ecf20Sopenharmony_ci 2598c2ecf20Sopenharmony_ci request.message_id = MSG_CONSOLE_GET_CLOCK_UID; 2608c2ecf20Sopenharmony_ci request.uid = get_console_mgr; 2618c2ecf20Sopenharmony_ci request.data = &get_console_mgr; 2628c2ecf20Sopenharmony_ci request.size = sizeof(get_console_mgr); 2638c2ecf20Sopenharmony_ci 2648c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(console_mgr), &console_mgr); 2658c2ecf20Sopenharmony_ci 2668c2ecf20Sopenharmony_ci if( (err < 0) || (console_mgr.error_code != 0) ) { 2678c2ecf20Sopenharmony_ci dev_dbg(&mgr->pci->dev, 2688c2ecf20Sopenharmony_ci "error MSG_CONSOLE_GET_CLOCK_UID : err=%x\n", 2698c2ecf20Sopenharmony_ci console_mgr.error_code); 2708c2ecf20Sopenharmony_ci return -EINVAL; 2718c2ecf20Sopenharmony_ci } 2728c2ecf20Sopenharmony_ci 2738c2ecf20Sopenharmony_ci /* used later for clock issues ! */ 2748c2ecf20Sopenharmony_ci mgr->uid_console_manager = console_mgr.uid; 2758c2ecf20Sopenharmony_ci 2768c2ecf20Sopenharmony_ci request.message_id = MSG_SYSTEM_ENUM_PHYSICAL_IO; 2778c2ecf20Sopenharmony_ci request.uid = (struct mixart_uid){0,0}; 2788c2ecf20Sopenharmony_ci request.data = &console_mgr.uid; 2798c2ecf20Sopenharmony_ci request.size = sizeof(console_mgr.uid); 2808c2ecf20Sopenharmony_ci 2818c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(phys_io), &phys_io); 2828c2ecf20Sopenharmony_ci if( (err < 0) || ( phys_io.error_code != 0 ) ) { 2838c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 2848c2ecf20Sopenharmony_ci "error MSG_SYSTEM_ENUM_PHYSICAL_IO err(%x) error_code(%x)\n", 2858c2ecf20Sopenharmony_ci err, phys_io.error_code); 2868c2ecf20Sopenharmony_ci return -EINVAL; 2878c2ecf20Sopenharmony_ci } 2888c2ecf20Sopenharmony_ci 2898c2ecf20Sopenharmony_ci /* min 2 phys io per card (analog in + analog out) */ 2908c2ecf20Sopenharmony_ci if (phys_io.nb_uid < MIXART_MAX_CARDS * 2) 2918c2ecf20Sopenharmony_ci return -EINVAL; 2928c2ecf20Sopenharmony_ci 2938c2ecf20Sopenharmony_ci for(k=0; k<mgr->num_cards; k++) { 2948c2ecf20Sopenharmony_ci mgr->chip[k]->uid_in_analog_physio = phys_io.uid[k]; 2958c2ecf20Sopenharmony_ci mgr->chip[k]->uid_out_analog_physio = phys_io.uid[phys_io.nb_uid/2 + k]; 2968c2ecf20Sopenharmony_ci } 2978c2ecf20Sopenharmony_ci 2988c2ecf20Sopenharmony_ci return 0; 2998c2ecf20Sopenharmony_ci} 3008c2ecf20Sopenharmony_ci 3018c2ecf20Sopenharmony_ci 3028c2ecf20Sopenharmony_cistatic int mixart_first_init(struct mixart_mgr *mgr) 3038c2ecf20Sopenharmony_ci{ 3048c2ecf20Sopenharmony_ci u32 k; 3058c2ecf20Sopenharmony_ci int err; 3068c2ecf20Sopenharmony_ci struct mixart_msg request; 3078c2ecf20Sopenharmony_ci 3088c2ecf20Sopenharmony_ci if((err = mixart_enum_connectors(mgr)) < 0) return err; 3098c2ecf20Sopenharmony_ci 3108c2ecf20Sopenharmony_ci if((err = mixart_enum_physio(mgr)) < 0) return err; 3118c2ecf20Sopenharmony_ci 3128c2ecf20Sopenharmony_ci /* send a synchro command to card (necessary to do this before first MSG_STREAM_START_STREAM_GRP_PACKET) */ 3138c2ecf20Sopenharmony_ci /* though why not here */ 3148c2ecf20Sopenharmony_ci request.message_id = MSG_SYSTEM_SEND_SYNCHRO_CMD; 3158c2ecf20Sopenharmony_ci request.uid = (struct mixart_uid){0,0}; 3168c2ecf20Sopenharmony_ci request.data = NULL; 3178c2ecf20Sopenharmony_ci request.size = 0; 3188c2ecf20Sopenharmony_ci /* this command has no data. response is a 32 bit status */ 3198c2ecf20Sopenharmony_ci err = snd_mixart_send_msg(mgr, &request, sizeof(k), &k); 3208c2ecf20Sopenharmony_ci if( (err < 0) || (k != 0) ) { 3218c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "error MSG_SYSTEM_SEND_SYNCHRO_CMD\n"); 3228c2ecf20Sopenharmony_ci return err == 0 ? -EINVAL : err; 3238c2ecf20Sopenharmony_ci } 3248c2ecf20Sopenharmony_ci 3258c2ecf20Sopenharmony_ci return 0; 3268c2ecf20Sopenharmony_ci} 3278c2ecf20Sopenharmony_ci 3288c2ecf20Sopenharmony_ci 3298c2ecf20Sopenharmony_ci/* firmware base addresses (when hard coded) */ 3308c2ecf20Sopenharmony_ci#define MIXART_MOTHERBOARD_XLX_BASE_ADDRESS 0x00600000 3318c2ecf20Sopenharmony_ci 3328c2ecf20Sopenharmony_cistatic int mixart_dsp_load(struct mixart_mgr* mgr, int index, const struct firmware *dsp) 3338c2ecf20Sopenharmony_ci{ 3348c2ecf20Sopenharmony_ci int err, card_index; 3358c2ecf20Sopenharmony_ci u32 status_xilinx, status_elf, status_daught; 3368c2ecf20Sopenharmony_ci u32 val; 3378c2ecf20Sopenharmony_ci 3388c2ecf20Sopenharmony_ci /* read motherboard xilinx status */ 3398c2ecf20Sopenharmony_ci status_xilinx = readl_be( MIXART_MEM( mgr,MIXART_PSEUDOREG_MXLX_STATUS_OFFSET )); 3408c2ecf20Sopenharmony_ci /* read elf status */ 3418c2ecf20Sopenharmony_ci status_elf = readl_be( MIXART_MEM( mgr,MIXART_PSEUDOREG_ELF_STATUS_OFFSET )); 3428c2ecf20Sopenharmony_ci /* read daughterboard xilinx status */ 3438c2ecf20Sopenharmony_ci status_daught = readl_be( MIXART_MEM( mgr,MIXART_PSEUDOREG_DXLX_STATUS_OFFSET )); 3448c2ecf20Sopenharmony_ci 3458c2ecf20Sopenharmony_ci /* motherboard xilinx status 5 will say that the board is performing a reset */ 3468c2ecf20Sopenharmony_ci if (status_xilinx == 5) { 3478c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "miXart is resetting !\n"); 3488c2ecf20Sopenharmony_ci return -EAGAIN; /* try again later */ 3498c2ecf20Sopenharmony_ci } 3508c2ecf20Sopenharmony_ci 3518c2ecf20Sopenharmony_ci switch (index) { 3528c2ecf20Sopenharmony_ci case MIXART_MOTHERBOARD_XLX_INDEX: 3538c2ecf20Sopenharmony_ci 3548c2ecf20Sopenharmony_ci /* xilinx already loaded ? */ 3558c2ecf20Sopenharmony_ci if (status_xilinx == 4) { 3568c2ecf20Sopenharmony_ci dev_dbg(&mgr->pci->dev, "xilinx is already loaded !\n"); 3578c2ecf20Sopenharmony_ci return 0; 3588c2ecf20Sopenharmony_ci } 3598c2ecf20Sopenharmony_ci /* the status should be 0 == "idle" */ 3608c2ecf20Sopenharmony_ci if (status_xilinx != 0) { 3618c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 3628c2ecf20Sopenharmony_ci "xilinx load error ! status = %d\n", 3638c2ecf20Sopenharmony_ci status_xilinx); 3648c2ecf20Sopenharmony_ci return -EIO; /* modprob -r may help ? */ 3658c2ecf20Sopenharmony_ci } 3668c2ecf20Sopenharmony_ci 3678c2ecf20Sopenharmony_ci /* check xilinx validity */ 3688c2ecf20Sopenharmony_ci if (((u32*)(dsp->data))[0] == 0xffffffff) 3698c2ecf20Sopenharmony_ci return -EINVAL; 3708c2ecf20Sopenharmony_ci if (dsp->size % 4) 3718c2ecf20Sopenharmony_ci return -EINVAL; 3728c2ecf20Sopenharmony_ci 3738c2ecf20Sopenharmony_ci /* set xilinx status to copying */ 3748c2ecf20Sopenharmony_ci writel_be( 1, MIXART_MEM( mgr, MIXART_PSEUDOREG_MXLX_STATUS_OFFSET )); 3758c2ecf20Sopenharmony_ci 3768c2ecf20Sopenharmony_ci /* setup xilinx base address */ 3778c2ecf20Sopenharmony_ci writel_be( MIXART_MOTHERBOARD_XLX_BASE_ADDRESS, MIXART_MEM( mgr,MIXART_PSEUDOREG_MXLX_BASE_ADDR_OFFSET )); 3788c2ecf20Sopenharmony_ci /* setup code size for xilinx file */ 3798c2ecf20Sopenharmony_ci writel_be( dsp->size, MIXART_MEM( mgr, MIXART_PSEUDOREG_MXLX_SIZE_OFFSET )); 3808c2ecf20Sopenharmony_ci 3818c2ecf20Sopenharmony_ci /* copy xilinx code */ 3828c2ecf20Sopenharmony_ci memcpy_toio( MIXART_MEM( mgr, MIXART_MOTHERBOARD_XLX_BASE_ADDRESS), dsp->data, dsp->size); 3838c2ecf20Sopenharmony_ci 3848c2ecf20Sopenharmony_ci /* set xilinx status to copy finished */ 3858c2ecf20Sopenharmony_ci writel_be( 2, MIXART_MEM( mgr, MIXART_PSEUDOREG_MXLX_STATUS_OFFSET )); 3868c2ecf20Sopenharmony_ci 3878c2ecf20Sopenharmony_ci /* return, because no further processing needed */ 3888c2ecf20Sopenharmony_ci return 0; 3898c2ecf20Sopenharmony_ci 3908c2ecf20Sopenharmony_ci case MIXART_MOTHERBOARD_ELF_INDEX: 3918c2ecf20Sopenharmony_ci 3928c2ecf20Sopenharmony_ci if (status_elf == 4) { 3938c2ecf20Sopenharmony_ci dev_dbg(&mgr->pci->dev, "elf file already loaded !\n"); 3948c2ecf20Sopenharmony_ci return 0; 3958c2ecf20Sopenharmony_ci } 3968c2ecf20Sopenharmony_ci 3978c2ecf20Sopenharmony_ci /* the status should be 0 == "idle" */ 3988c2ecf20Sopenharmony_ci if (status_elf != 0) { 3998c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 4008c2ecf20Sopenharmony_ci "elf load error ! status = %d\n", 4018c2ecf20Sopenharmony_ci status_elf); 4028c2ecf20Sopenharmony_ci return -EIO; /* modprob -r may help ? */ 4038c2ecf20Sopenharmony_ci } 4048c2ecf20Sopenharmony_ci 4058c2ecf20Sopenharmony_ci /* wait for xilinx status == 4 */ 4068c2ecf20Sopenharmony_ci err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_MXLX_STATUS_OFFSET, 1, 4, 500); /* 5sec */ 4078c2ecf20Sopenharmony_ci if (err < 0) { 4088c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "xilinx was not loaded or " 4098c2ecf20Sopenharmony_ci "could not be started\n"); 4108c2ecf20Sopenharmony_ci return err; 4118c2ecf20Sopenharmony_ci } 4128c2ecf20Sopenharmony_ci 4138c2ecf20Sopenharmony_ci /* init some data on the card */ 4148c2ecf20Sopenharmony_ci writel_be( 0, MIXART_MEM( mgr, MIXART_PSEUDOREG_BOARDNUMBER ) ); /* set miXart boardnumber to 0 */ 4158c2ecf20Sopenharmony_ci writel_be( 0, MIXART_MEM( mgr, MIXART_FLOWTABLE_PTR ) ); /* reset pointer to flow table on miXart */ 4168c2ecf20Sopenharmony_ci 4178c2ecf20Sopenharmony_ci /* set elf status to copying */ 4188c2ecf20Sopenharmony_ci writel_be( 1, MIXART_MEM( mgr, MIXART_PSEUDOREG_ELF_STATUS_OFFSET )); 4198c2ecf20Sopenharmony_ci 4208c2ecf20Sopenharmony_ci /* process the copying of the elf packets */ 4218c2ecf20Sopenharmony_ci err = mixart_load_elf( mgr, dsp ); 4228c2ecf20Sopenharmony_ci if (err < 0) return err; 4238c2ecf20Sopenharmony_ci 4248c2ecf20Sopenharmony_ci /* set elf status to copy finished */ 4258c2ecf20Sopenharmony_ci writel_be( 2, MIXART_MEM( mgr, MIXART_PSEUDOREG_ELF_STATUS_OFFSET )); 4268c2ecf20Sopenharmony_ci 4278c2ecf20Sopenharmony_ci /* wait for elf status == 4 */ 4288c2ecf20Sopenharmony_ci err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_ELF_STATUS_OFFSET, 1, 4, 300); /* 3sec */ 4298c2ecf20Sopenharmony_ci if (err < 0) { 4308c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "elf could not be started\n"); 4318c2ecf20Sopenharmony_ci return err; 4328c2ecf20Sopenharmony_ci } 4338c2ecf20Sopenharmony_ci 4348c2ecf20Sopenharmony_ci /* miXart waits at this point on the pointer to the flow table */ 4358c2ecf20Sopenharmony_ci writel_be( (u32)mgr->flowinfo.addr, MIXART_MEM( mgr, MIXART_FLOWTABLE_PTR ) ); /* give pointer of flow table to miXart */ 4368c2ecf20Sopenharmony_ci 4378c2ecf20Sopenharmony_ci return 0; /* return, another xilinx file has to be loaded before */ 4388c2ecf20Sopenharmony_ci 4398c2ecf20Sopenharmony_ci case MIXART_AESEBUBOARD_XLX_INDEX: 4408c2ecf20Sopenharmony_ci default: 4418c2ecf20Sopenharmony_ci 4428c2ecf20Sopenharmony_ci /* elf and xilinx should be loaded */ 4438c2ecf20Sopenharmony_ci if (status_elf != 4 || status_xilinx != 4) { 4448c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "xilinx or elf not " 4458c2ecf20Sopenharmony_ci "successfully loaded\n"); 4468c2ecf20Sopenharmony_ci return -EIO; /* modprob -r may help ? */ 4478c2ecf20Sopenharmony_ci } 4488c2ecf20Sopenharmony_ci 4498c2ecf20Sopenharmony_ci /* wait for daughter detection != 0 */ 4508c2ecf20Sopenharmony_ci err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_DBRD_PRESENCE_OFFSET, 0, 0, 30); /* 300msec */ 4518c2ecf20Sopenharmony_ci if (err < 0) { 4528c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "error starting elf file\n"); 4538c2ecf20Sopenharmony_ci return err; 4548c2ecf20Sopenharmony_ci } 4558c2ecf20Sopenharmony_ci 4568c2ecf20Sopenharmony_ci /* the board type can now be retrieved */ 4578c2ecf20Sopenharmony_ci mgr->board_type = (DAUGHTER_TYPE_MASK & readl_be( MIXART_MEM( mgr, MIXART_PSEUDOREG_DBRD_TYPE_OFFSET))); 4588c2ecf20Sopenharmony_ci 4598c2ecf20Sopenharmony_ci if (mgr->board_type == MIXART_DAUGHTER_TYPE_NONE) 4608c2ecf20Sopenharmony_ci break; /* no daughter board; the file does not have to be loaded, continue after the switch */ 4618c2ecf20Sopenharmony_ci 4628c2ecf20Sopenharmony_ci /* only if aesebu daughter board presence (elf code must run) */ 4638c2ecf20Sopenharmony_ci if (mgr->board_type != MIXART_DAUGHTER_TYPE_AES ) 4648c2ecf20Sopenharmony_ci return -EINVAL; 4658c2ecf20Sopenharmony_ci 4668c2ecf20Sopenharmony_ci /* daughter should be idle */ 4678c2ecf20Sopenharmony_ci if (status_daught != 0) { 4688c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 4698c2ecf20Sopenharmony_ci "daughter load error ! status = %d\n", 4708c2ecf20Sopenharmony_ci status_daught); 4718c2ecf20Sopenharmony_ci return -EIO; /* modprob -r may help ? */ 4728c2ecf20Sopenharmony_ci } 4738c2ecf20Sopenharmony_ci 4748c2ecf20Sopenharmony_ci /* check daughterboard xilinx validity */ 4758c2ecf20Sopenharmony_ci if (((u32*)(dsp->data))[0] == 0xffffffff) 4768c2ecf20Sopenharmony_ci return -EINVAL; 4778c2ecf20Sopenharmony_ci if (dsp->size % 4) 4788c2ecf20Sopenharmony_ci return -EINVAL; 4798c2ecf20Sopenharmony_ci 4808c2ecf20Sopenharmony_ci /* inform mixart about the size of the file */ 4818c2ecf20Sopenharmony_ci writel_be( dsp->size, MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_SIZE_OFFSET )); 4828c2ecf20Sopenharmony_ci 4838c2ecf20Sopenharmony_ci /* set daughterboard status to 1 */ 4848c2ecf20Sopenharmony_ci writel_be( 1, MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET )); 4858c2ecf20Sopenharmony_ci 4868c2ecf20Sopenharmony_ci /* wait for status == 2 */ 4878c2ecf20Sopenharmony_ci err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET, 1, 2, 30); /* 300msec */ 4888c2ecf20Sopenharmony_ci if (err < 0) { 4898c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "daughter board load error\n"); 4908c2ecf20Sopenharmony_ci return err; 4918c2ecf20Sopenharmony_ci } 4928c2ecf20Sopenharmony_ci 4938c2ecf20Sopenharmony_ci /* get the address where to write the file */ 4948c2ecf20Sopenharmony_ci val = readl_be( MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_BASE_ADDR_OFFSET )); 4958c2ecf20Sopenharmony_ci if (!val) 4968c2ecf20Sopenharmony_ci return -EINVAL; 4978c2ecf20Sopenharmony_ci 4988c2ecf20Sopenharmony_ci /* copy daughterboard xilinx code */ 4998c2ecf20Sopenharmony_ci memcpy_toio( MIXART_MEM( mgr, val), dsp->data, dsp->size); 5008c2ecf20Sopenharmony_ci 5018c2ecf20Sopenharmony_ci /* set daughterboard status to 4 */ 5028c2ecf20Sopenharmony_ci writel_be( 4, MIXART_MEM( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET )); 5038c2ecf20Sopenharmony_ci 5048c2ecf20Sopenharmony_ci /* continue with init */ 5058c2ecf20Sopenharmony_ci break; 5068c2ecf20Sopenharmony_ci } /* end of switch file index*/ 5078c2ecf20Sopenharmony_ci 5088c2ecf20Sopenharmony_ci /* wait for daughter status == 3 */ 5098c2ecf20Sopenharmony_ci err = mixart_wait_nice_for_register_value( mgr, MIXART_PSEUDOREG_DXLX_STATUS_OFFSET, 1, 3, 300); /* 3sec */ 5108c2ecf20Sopenharmony_ci if (err < 0) { 5118c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 5128c2ecf20Sopenharmony_ci "daughter board could not be initialised\n"); 5138c2ecf20Sopenharmony_ci return err; 5148c2ecf20Sopenharmony_ci } 5158c2ecf20Sopenharmony_ci 5168c2ecf20Sopenharmony_ci /* init mailbox (communication with embedded) */ 5178c2ecf20Sopenharmony_ci snd_mixart_init_mailbox(mgr); 5188c2ecf20Sopenharmony_ci 5198c2ecf20Sopenharmony_ci /* first communication with embedded */ 5208c2ecf20Sopenharmony_ci err = mixart_first_init(mgr); 5218c2ecf20Sopenharmony_ci if (err < 0) { 5228c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, "miXart could not be set up\n"); 5238c2ecf20Sopenharmony_ci return err; 5248c2ecf20Sopenharmony_ci } 5258c2ecf20Sopenharmony_ci 5268c2ecf20Sopenharmony_ci /* create devices and mixer in accordance with HW options*/ 5278c2ecf20Sopenharmony_ci for (card_index = 0; card_index < mgr->num_cards; card_index++) { 5288c2ecf20Sopenharmony_ci struct snd_mixart *chip = mgr->chip[card_index]; 5298c2ecf20Sopenharmony_ci 5308c2ecf20Sopenharmony_ci if ((err = snd_mixart_create_pcm(chip)) < 0) 5318c2ecf20Sopenharmony_ci return err; 5328c2ecf20Sopenharmony_ci 5338c2ecf20Sopenharmony_ci if (card_index == 0) { 5348c2ecf20Sopenharmony_ci if ((err = snd_mixart_create_mixer(chip->mgr)) < 0) 5358c2ecf20Sopenharmony_ci return err; 5368c2ecf20Sopenharmony_ci } 5378c2ecf20Sopenharmony_ci 5388c2ecf20Sopenharmony_ci if ((err = snd_card_register(chip->card)) < 0) 5398c2ecf20Sopenharmony_ci return err; 5408c2ecf20Sopenharmony_ci } 5418c2ecf20Sopenharmony_ci 5428c2ecf20Sopenharmony_ci dev_dbg(&mgr->pci->dev, 5438c2ecf20Sopenharmony_ci "miXart firmware downloaded and successfully set up\n"); 5448c2ecf20Sopenharmony_ci 5458c2ecf20Sopenharmony_ci return 0; 5468c2ecf20Sopenharmony_ci} 5478c2ecf20Sopenharmony_ci 5488c2ecf20Sopenharmony_ci 5498c2ecf20Sopenharmony_ciint snd_mixart_setup_firmware(struct mixart_mgr *mgr) 5508c2ecf20Sopenharmony_ci{ 5518c2ecf20Sopenharmony_ci static const char * const fw_files[3] = { 5528c2ecf20Sopenharmony_ci "miXart8.xlx", "miXart8.elf", "miXart8AES.xlx" 5538c2ecf20Sopenharmony_ci }; 5548c2ecf20Sopenharmony_ci char path[32]; 5558c2ecf20Sopenharmony_ci 5568c2ecf20Sopenharmony_ci const struct firmware *fw_entry; 5578c2ecf20Sopenharmony_ci int i, err; 5588c2ecf20Sopenharmony_ci 5598c2ecf20Sopenharmony_ci for (i = 0; i < 3; i++) { 5608c2ecf20Sopenharmony_ci sprintf(path, "mixart/%s", fw_files[i]); 5618c2ecf20Sopenharmony_ci if (request_firmware(&fw_entry, path, &mgr->pci->dev)) { 5628c2ecf20Sopenharmony_ci dev_err(&mgr->pci->dev, 5638c2ecf20Sopenharmony_ci "miXart: can't load firmware %s\n", path); 5648c2ecf20Sopenharmony_ci return -ENOENT; 5658c2ecf20Sopenharmony_ci } 5668c2ecf20Sopenharmony_ci /* fake hwdep dsp record */ 5678c2ecf20Sopenharmony_ci err = mixart_dsp_load(mgr, i, fw_entry); 5688c2ecf20Sopenharmony_ci release_firmware(fw_entry); 5698c2ecf20Sopenharmony_ci if (err < 0) 5708c2ecf20Sopenharmony_ci return err; 5718c2ecf20Sopenharmony_ci mgr->dsp_loaded |= 1 << i; 5728c2ecf20Sopenharmony_ci } 5738c2ecf20Sopenharmony_ci return 0; 5748c2ecf20Sopenharmony_ci} 5758c2ecf20Sopenharmony_ci 5768c2ecf20Sopenharmony_ciMODULE_FIRMWARE("mixart/miXart8.xlx"); 5778c2ecf20Sopenharmony_ciMODULE_FIRMWARE("mixart/miXart8.elf"); 5788c2ecf20Sopenharmony_ciMODULE_FIRMWARE("mixart/miXart8AES.xlx"); 579