summaryrefslogtreecommitdiff
path: root/sound/soc/sof/intel/hda-bus.c
blob: a7e6d8227df6f45727b8f2dfcd911df700bdbe67 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
// SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause)
//
// This file is provided under a dual BSD/GPLv2 license.  When using or
// redistributing this file, you may do so under either license.
//
// Copyright(c) 2018 Intel Corporation. All rights reserved.
//
// Authors: Keyon Jie <yang.jie@linux.intel.com>

#include <linux/io.h>
#include <sound/hdaudio.h>
#include "../sof-priv.h"
#include "hda.h"

#if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA)

static const struct hdac_bus_ops bus_ops = {
	.command = snd_hdac_bus_send_cmd,
	.get_response = snd_hdac_bus_get_response,
};

#endif

static void sof_hda_writel(u32 value, u32 __iomem *addr)
{
	writel(value, addr);
}

static u32 sof_hda_readl(u32 __iomem *addr)
{
	return readl(addr);
}

static void sof_hda_writew(u16 value, u16 __iomem *addr)
{
	writew(value, addr);
}

static u16 sof_hda_readw(u16 __iomem *addr)
{
	return readw(addr);
}

static void sof_hda_writeb(u8 value, u8 __iomem *addr)
{
	writeb(value, addr);
}

static u8 sof_hda_readb(u8 __iomem *addr)
{
	return readb(addr);
}

static int sof_hda_dma_alloc_pages(struct hdac_bus *bus, int type,
				   size_t size, struct snd_dma_buffer *buf)
{
	return snd_dma_alloc_pages(type, bus->dev, size, buf);
}

static void sof_hda_dma_free_pages(struct hdac_bus *bus,
				   struct snd_dma_buffer *buf)
{
	snd_dma_free_pages(buf);
}

static const struct hdac_io_ops io_ops = {
	.reg_writel = sof_hda_writel,
	.reg_readl = sof_hda_readl,
	.reg_writew = sof_hda_writew,
	.reg_readw = sof_hda_readw,
	.reg_writeb = sof_hda_writeb,
	.reg_readb = sof_hda_readb,
	.dma_alloc_pages = sof_hda_dma_alloc_pages,
	.dma_free_pages = sof_hda_dma_free_pages,
};

/*
 * This can be used for both with/without hda link support.
 */
void sof_hda_bus_init(struct hdac_bus *bus, struct device *dev,
		      const struct hdac_ext_bus_ops *ext_ops)
{
	memset(bus, 0, sizeof(*bus));
	bus->dev = dev;

	bus->io_ops = &io_ops;
	INIT_LIST_HEAD(&bus->stream_list);

	bus->irq = -1;
	bus->ext_ops = ext_ops;

	/*
	 * There is only one HDA bus atm. keep the index as 0.
	 * Need to fix when there are more than one HDA bus.
	 */
	bus->idx = 0;

	spin_lock_init(&bus->reg_lock);

#if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA)
	INIT_LIST_HEAD(&bus->codec_list);
	INIT_LIST_HEAD(&bus->hlink_list);

	mutex_init(&bus->cmd_mutex);
	mutex_init(&bus->lock);
	bus->ops = &bus_ops;
	INIT_WORK(&bus->unsol_work, snd_hdac_bus_process_unsol_events);
	bus->cmd_dma_state = true;
#endif

}