1d7b50c0cSAndreas Färber /*
2d7b50c0cSAndreas Färber * QTest testcase for AC97
3d7b50c0cSAndreas Färber *
4d7b50c0cSAndreas Färber * Copyright (c) 2014 SUSE LINUX Products GmbH
5d7b50c0cSAndreas Färber *
6d7b50c0cSAndreas Färber * This work is licensed under the terms of the GNU GPL, version 2 or later.
7d7b50c0cSAndreas Färber * See the COPYING file in the top-level directory.
8d7b50c0cSAndreas Färber */
9d7b50c0cSAndreas Färber
10d7b50c0cSAndreas Färber #include "qemu/osdep.h"
11907b5105SMarc-André Lureau #include "libqtest.h"
120b8fa32fSMarkus Armbruster #include "qemu/module.h"
1386dc943fSEmanuele Giuseppe Esposito #include "libqos/qgraph.h"
1486dc943fSEmanuele Giuseppe Esposito #include "libqos/pci.h"
15d7b50c0cSAndreas Färber
1686dc943fSEmanuele Giuseppe Esposito typedef struct QAC97 QAC97;
1786dc943fSEmanuele Giuseppe Esposito
1886dc943fSEmanuele Giuseppe Esposito struct QAC97 {
1986dc943fSEmanuele Giuseppe Esposito QOSGraphObject obj;
2086dc943fSEmanuele Giuseppe Esposito QPCIDevice dev;
2186dc943fSEmanuele Giuseppe Esposito };
2286dc943fSEmanuele Giuseppe Esposito
ac97_get_driver(void * obj,const char * interface)2386dc943fSEmanuele Giuseppe Esposito static void *ac97_get_driver(void *obj, const char *interface)
24d7b50c0cSAndreas Färber {
2586dc943fSEmanuele Giuseppe Esposito QAC97 *ac97 = obj;
2686dc943fSEmanuele Giuseppe Esposito
2786dc943fSEmanuele Giuseppe Esposito if (!g_strcmp0(interface, "pci-device")) {
2886dc943fSEmanuele Giuseppe Esposito return &ac97->dev;
29d7b50c0cSAndreas Färber }
30d7b50c0cSAndreas Färber
31146f39d7SAkihiko Odaki fprintf(stderr, "%s not present in ac97\n", interface);
3286dc943fSEmanuele Giuseppe Esposito g_assert_not_reached();
33d7b50c0cSAndreas Färber }
3486dc943fSEmanuele Giuseppe Esposito
ac97_create(void * pci_bus,QGuestAllocator * alloc,void * addr)3586dc943fSEmanuele Giuseppe Esposito static void *ac97_create(void *pci_bus, QGuestAllocator *alloc, void *addr)
3686dc943fSEmanuele Giuseppe Esposito {
3786dc943fSEmanuele Giuseppe Esposito QAC97 *ac97 = g_new0(QAC97, 1);
3886dc943fSEmanuele Giuseppe Esposito QPCIBus *bus = pci_bus;
3986dc943fSEmanuele Giuseppe Esposito
4086dc943fSEmanuele Giuseppe Esposito qpci_device_init(&ac97->dev, bus, addr);
4186dc943fSEmanuele Giuseppe Esposito ac97->obj.get_driver = ac97_get_driver;
4286dc943fSEmanuele Giuseppe Esposito return &ac97->obj;
4386dc943fSEmanuele Giuseppe Esposito }
4486dc943fSEmanuele Giuseppe Esposito
45*031616cdSVolker Rümelin /*
46*031616cdSVolker Rümelin * This is rather a test of the audio subsystem and not an AC97 test. Test if
47*031616cdSVolker Rümelin * the audio subsystem can handle a 44100/1 upsample ratio. For some time this
48*031616cdSVolker Rümelin * used to trigger QEMU aborts.
49*031616cdSVolker Rümelin */
ac97_playback_upsample(void * obj,void * data,QGuestAllocator * alloc)50*031616cdSVolker Rümelin static void ac97_playback_upsample(void *obj, void *data, QGuestAllocator *alloc)
51*031616cdSVolker Rümelin {
52*031616cdSVolker Rümelin QAC97 *ac97 = obj;
53*031616cdSVolker Rümelin QPCIDevice *dev = &ac97->dev;
54*031616cdSVolker Rümelin QPCIBar bar0;
55*031616cdSVolker Rümelin
56*031616cdSVolker Rümelin qpci_device_enable(dev);
57*031616cdSVolker Rümelin bar0 = qpci_iomap(dev, 0, NULL);
58*031616cdSVolker Rümelin /* IOBAR0 offset 0x2c: PCM Front DAC Rate */
59*031616cdSVolker Rümelin qpci_io_writew(dev, bar0, 0x2c, 0x1);
60*031616cdSVolker Rümelin }
61*031616cdSVolker Rümelin
62*031616cdSVolker Rümelin /*
63*031616cdSVolker Rümelin * This test is similar to the playback upsample test. QEMU shouldn't abort if
64*031616cdSVolker Rümelin * asked for a 1/44100 downsample ratio.
65*031616cdSVolker Rümelin */
ac97_record_downsample(void * obj,void * data,QGuestAllocator * alloc)66*031616cdSVolker Rümelin static void ac97_record_downsample(void *obj, void *data, QGuestAllocator *alloc)
67*031616cdSVolker Rümelin {
68*031616cdSVolker Rümelin QAC97 *ac97 = obj;
69*031616cdSVolker Rümelin QPCIDevice *dev = &ac97->dev;
70*031616cdSVolker Rümelin QPCIBar bar0;
71*031616cdSVolker Rümelin
72*031616cdSVolker Rümelin qpci_device_enable(dev);
73*031616cdSVolker Rümelin bar0 = qpci_iomap(dev, 0, NULL);
74*031616cdSVolker Rümelin /* IOBAR0 offset 0x32: PCM L/R ADC Rate */
75*031616cdSVolker Rümelin qpci_io_writew(dev, bar0, 0x32, 0x1);
76*031616cdSVolker Rümelin }
77*031616cdSVolker Rümelin
ac97_register_nodes(void)7886dc943fSEmanuele Giuseppe Esposito static void ac97_register_nodes(void)
7986dc943fSEmanuele Giuseppe Esposito {
8086dc943fSEmanuele Giuseppe Esposito QOSGraphEdgeOptions opts = {
81*031616cdSVolker Rümelin .extra_device_opts = "addr=04.0,audiodev=snd0",
82*031616cdSVolker Rümelin .after_cmd_line = "-audiodev none,id=snd0"
83*031616cdSVolker Rümelin ",out.frequency=44100,in.frequency=44100",
8486dc943fSEmanuele Giuseppe Esposito };
8586dc943fSEmanuele Giuseppe Esposito add_qpci_address(&opts, &(QPCIAddress) { .devfn = QPCI_DEVFN(4, 0) });
8686dc943fSEmanuele Giuseppe Esposito
8786dc943fSEmanuele Giuseppe Esposito qos_node_create_driver("AC97", ac97_create);
8886dc943fSEmanuele Giuseppe Esposito qos_node_produces("AC97", "pci-device");
8986dc943fSEmanuele Giuseppe Esposito qos_node_consumes("AC97", "pci-bus", &opts);
90*031616cdSVolker Rümelin
91*031616cdSVolker Rümelin qos_add_test("playback_upsample", "AC97", ac97_playback_upsample, NULL);
92*031616cdSVolker Rümelin qos_add_test("record_downsample", "AC97", ac97_record_downsample, NULL);
9386dc943fSEmanuele Giuseppe Esposito }
9486dc943fSEmanuele Giuseppe Esposito
9586dc943fSEmanuele Giuseppe Esposito libqos_init(ac97_register_nodes);
96