162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *  Generic driver for AD1848/AD1847/CS4248 chips (0.1 Alpha)
462306a36Sopenharmony_ci *  Copyright (c) by Tugrul Galatali <galatalt@stuy.edu>,
562306a36Sopenharmony_ci *                   Jaroslav Kysela <perex@perex.cz>
662306a36Sopenharmony_ci *  Based on card-4232.c by Jaroslav Kysela <perex@perex.cz>
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/init.h>
1062306a36Sopenharmony_ci#include <linux/err.h>
1162306a36Sopenharmony_ci#include <linux/isa.h>
1262306a36Sopenharmony_ci#include <linux/time.h>
1362306a36Sopenharmony_ci#include <linux/wait.h>
1462306a36Sopenharmony_ci#include <linux/module.h>
1562306a36Sopenharmony_ci#include <sound/core.h>
1662306a36Sopenharmony_ci#include <sound/wss.h>
1762306a36Sopenharmony_ci#include <sound/initval.h>
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci#define CRD_NAME "Generic AD1848/AD1847/CS4248"
2062306a36Sopenharmony_ci#define DEV_NAME "ad1848"
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_ciMODULE_DESCRIPTION(CRD_NAME);
2362306a36Sopenharmony_ciMODULE_AUTHOR("Tugrul Galatali <galatalt@stuy.edu>, Jaroslav Kysela <perex@perex.cz>");
2462306a36Sopenharmony_ciMODULE_LICENSE("GPL");
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_cistatic int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX;	/* Index 0-MAX */
2762306a36Sopenharmony_cistatic char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR;	/* ID for this card */
2862306a36Sopenharmony_cistatic bool enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE;	/* Enable this card */
2962306a36Sopenharmony_cistatic long port[SNDRV_CARDS] = SNDRV_DEFAULT_PORT;	/* PnP setup */
3062306a36Sopenharmony_cistatic int irq[SNDRV_CARDS] = SNDRV_DEFAULT_IRQ;	/* 5,7,9,11,12,15 */
3162306a36Sopenharmony_cistatic int dma1[SNDRV_CARDS] = SNDRV_DEFAULT_DMA;	/* 0,1,3,5,6,7 */
3262306a36Sopenharmony_cistatic bool thinkpad[SNDRV_CARDS];			/* Thinkpad special case */
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_cimodule_param_array(index, int, NULL, 0444);
3562306a36Sopenharmony_ciMODULE_PARM_DESC(index, "Index value for " CRD_NAME " soundcard.");
3662306a36Sopenharmony_cimodule_param_array(id, charp, NULL, 0444);
3762306a36Sopenharmony_ciMODULE_PARM_DESC(id, "ID string for " CRD_NAME " soundcard.");
3862306a36Sopenharmony_cimodule_param_array(enable, bool, NULL, 0444);
3962306a36Sopenharmony_ciMODULE_PARM_DESC(enable, "Enable " CRD_NAME " soundcard.");
4062306a36Sopenharmony_cimodule_param_hw_array(port, long, ioport, NULL, 0444);
4162306a36Sopenharmony_ciMODULE_PARM_DESC(port, "Port # for " CRD_NAME " driver.");
4262306a36Sopenharmony_cimodule_param_hw_array(irq, int, irq, NULL, 0444);
4362306a36Sopenharmony_ciMODULE_PARM_DESC(irq, "IRQ # for " CRD_NAME " driver.");
4462306a36Sopenharmony_cimodule_param_hw_array(dma1, int, dma, NULL, 0444);
4562306a36Sopenharmony_ciMODULE_PARM_DESC(dma1, "DMA1 # for " CRD_NAME " driver.");
4662306a36Sopenharmony_cimodule_param_array(thinkpad, bool, NULL, 0444);
4762306a36Sopenharmony_ciMODULE_PARM_DESC(thinkpad, "Enable only for the onboard CS4248 of IBM Thinkpad 360/750/755 series.");
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_cistatic int snd_ad1848_match(struct device *dev, unsigned int n)
5062306a36Sopenharmony_ci{
5162306a36Sopenharmony_ci	if (!enable[n])
5262306a36Sopenharmony_ci		return 0;
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci	if (port[n] == SNDRV_AUTO_PORT) {
5562306a36Sopenharmony_ci		dev_err(dev, "please specify port\n");
5662306a36Sopenharmony_ci		return 0;
5762306a36Sopenharmony_ci	}
5862306a36Sopenharmony_ci	if (irq[n] == SNDRV_AUTO_IRQ) {
5962306a36Sopenharmony_ci		dev_err(dev, "please specify irq\n");
6062306a36Sopenharmony_ci		return 0;
6162306a36Sopenharmony_ci	}
6262306a36Sopenharmony_ci	if (dma1[n] == SNDRV_AUTO_DMA) {
6362306a36Sopenharmony_ci		dev_err(dev, "please specify dma1\n");
6462306a36Sopenharmony_ci		return 0;
6562306a36Sopenharmony_ci	}
6662306a36Sopenharmony_ci	return 1;
6762306a36Sopenharmony_ci}
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_cistatic int snd_ad1848_probe(struct device *dev, unsigned int n)
7062306a36Sopenharmony_ci{
7162306a36Sopenharmony_ci	struct snd_card *card;
7262306a36Sopenharmony_ci	struct snd_wss *chip;
7362306a36Sopenharmony_ci	int error;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	error = snd_devm_card_new(dev, index[n], id[n], THIS_MODULE, 0, &card);
7662306a36Sopenharmony_ci	if (error < 0)
7762306a36Sopenharmony_ci		return error;
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci	error = snd_wss_create(card, port[n], -1, irq[n], dma1[n], -1,
8062306a36Sopenharmony_ci			thinkpad[n] ? WSS_HW_THINKPAD : WSS_HW_DETECT,
8162306a36Sopenharmony_ci			0, &chip);
8262306a36Sopenharmony_ci	if (error < 0)
8362306a36Sopenharmony_ci		return error;
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_ci	card->private_data = chip;
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	error = snd_wss_pcm(chip, 0);
8862306a36Sopenharmony_ci	if (error < 0)
8962306a36Sopenharmony_ci		return error;
9062306a36Sopenharmony_ci
9162306a36Sopenharmony_ci	error = snd_wss_mixer(chip);
9262306a36Sopenharmony_ci	if (error < 0)
9362306a36Sopenharmony_ci		return error;
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_ci	strscpy(card->driver, "AD1848", sizeof(card->driver));
9662306a36Sopenharmony_ci	strscpy(card->shortname, chip->pcm->name, sizeof(card->shortname));
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ci	if (!thinkpad[n])
9962306a36Sopenharmony_ci		scnprintf(card->longname, sizeof(card->longname),
10062306a36Sopenharmony_ci			  "%s at 0x%lx, irq %d, dma %d",
10162306a36Sopenharmony_ci			  chip->pcm->name, chip->port, irq[n], dma1[n]);
10262306a36Sopenharmony_ci	else
10362306a36Sopenharmony_ci		scnprintf(card->longname, sizeof(card->longname),
10462306a36Sopenharmony_ci			  "%s at 0x%lx, irq %d, dma %d [Thinkpad]",
10562306a36Sopenharmony_ci			  chip->pcm->name, chip->port, irq[n], dma1[n]);
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci	error = snd_card_register(card);
10862306a36Sopenharmony_ci	if (error < 0)
10962306a36Sopenharmony_ci		return error;
11062306a36Sopenharmony_ci
11162306a36Sopenharmony_ci	dev_set_drvdata(dev, card);
11262306a36Sopenharmony_ci	return 0;
11362306a36Sopenharmony_ci}
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci#ifdef CONFIG_PM
11662306a36Sopenharmony_cistatic int snd_ad1848_suspend(struct device *dev, unsigned int n, pm_message_t state)
11762306a36Sopenharmony_ci{
11862306a36Sopenharmony_ci	struct snd_card *card = dev_get_drvdata(dev);
11962306a36Sopenharmony_ci	struct snd_wss *chip = card->private_data;
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	snd_power_change_state(card, SNDRV_CTL_POWER_D3hot);
12262306a36Sopenharmony_ci	chip->suspend(chip);
12362306a36Sopenharmony_ci	return 0;
12462306a36Sopenharmony_ci}
12562306a36Sopenharmony_ci
12662306a36Sopenharmony_cistatic int snd_ad1848_resume(struct device *dev, unsigned int n)
12762306a36Sopenharmony_ci{
12862306a36Sopenharmony_ci	struct snd_card *card = dev_get_drvdata(dev);
12962306a36Sopenharmony_ci	struct snd_wss *chip = card->private_data;
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	chip->resume(chip);
13262306a36Sopenharmony_ci	snd_power_change_state(card, SNDRV_CTL_POWER_D0);
13362306a36Sopenharmony_ci	return 0;
13462306a36Sopenharmony_ci}
13562306a36Sopenharmony_ci#endif
13662306a36Sopenharmony_ci
13762306a36Sopenharmony_cistatic struct isa_driver snd_ad1848_driver = {
13862306a36Sopenharmony_ci	.match		= snd_ad1848_match,
13962306a36Sopenharmony_ci	.probe		= snd_ad1848_probe,
14062306a36Sopenharmony_ci#ifdef CONFIG_PM
14162306a36Sopenharmony_ci	.suspend	= snd_ad1848_suspend,
14262306a36Sopenharmony_ci	.resume		= snd_ad1848_resume,
14362306a36Sopenharmony_ci#endif
14462306a36Sopenharmony_ci	.driver		= {
14562306a36Sopenharmony_ci		.name	= DEV_NAME
14662306a36Sopenharmony_ci	}
14762306a36Sopenharmony_ci};
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_cimodule_isa_driver(snd_ad1848_driver, SNDRV_CARDS);
150