xref: /aosp_15_r20/external/tinyalsa_new/tests/src/pcm_test.cc (revision 02e95f1a335b55495d41ca67eaf42361f13704fa)
1*02e95f1aSMarcin Radomski /* pcm_out_test.c
2*02e95f1aSMarcin Radomski **
3*02e95f1aSMarcin Radomski ** Copyright 2020, The Android Open Source Project
4*02e95f1aSMarcin Radomski **
5*02e95f1aSMarcin Radomski ** Redistribution and use in source and binary forms, with or without
6*02e95f1aSMarcin Radomski ** modification, are permitted provided that the following conditions are met:
7*02e95f1aSMarcin Radomski **     * Redistributions of source code must retain the above copyright
8*02e95f1aSMarcin Radomski **       notice, this list of conditions and the following disclaimer.
9*02e95f1aSMarcin Radomski **     * Redistributions in binary form must reproduce the above copyright
10*02e95f1aSMarcin Radomski **       notice, this list of conditions and the following disclaimer in the
11*02e95f1aSMarcin Radomski **       documentation and/or other materials provided with the distribution.
12*02e95f1aSMarcin Radomski **     * Neither the name of The Android Open Source Project nor the names of
13*02e95f1aSMarcin Radomski **       its contributors may be used to endorse or promote products derived
14*02e95f1aSMarcin Radomski **       from this software without specific prior written permission.
15*02e95f1aSMarcin Radomski **
16*02e95f1aSMarcin Radomski ** THIS SOFTWARE IS PROVIDED BY The Android Open Source Project ``AS IS'' AND
17*02e95f1aSMarcin Radomski ** ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*02e95f1aSMarcin Radomski ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*02e95f1aSMarcin Radomski ** ARE DISCLAIMED. IN NO EVENT SHALL The Android Open Source Project BE LIABLE
20*02e95f1aSMarcin Radomski ** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21*02e95f1aSMarcin Radomski ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22*02e95f1aSMarcin Radomski ** SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
23*02e95f1aSMarcin Radomski ** CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24*02e95f1aSMarcin Radomski ** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25*02e95f1aSMarcin Radomski ** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
26*02e95f1aSMarcin Radomski ** DAMAGE.
27*02e95f1aSMarcin Radomski */
28*02e95f1aSMarcin Radomski 
29*02e95f1aSMarcin Radomski #include <cstdio>
30*02e95f1aSMarcin Radomski #include <fstream>
31*02e95f1aSMarcin Radomski #include <iostream>
32*02e95f1aSMarcin Radomski #include <memory>
33*02e95f1aSMarcin Radomski #include <string_view>
34*02e95f1aSMarcin Radomski #include <string>
35*02e95f1aSMarcin Radomski #include <thread>
36*02e95f1aSMarcin Radomski 
37*02e95f1aSMarcin Radomski #include <gtest/gtest.h>
38*02e95f1aSMarcin Radomski 
39*02e95f1aSMarcin Radomski #include "tinyalsa/pcm.h"
40*02e95f1aSMarcin Radomski 
41*02e95f1aSMarcin Radomski #include "pcm_test_device.h"
42*02e95f1aSMarcin Radomski 
43*02e95f1aSMarcin Radomski namespace tinyalsa {
44*02e95f1aSMarcin Radomski namespace testing {
45*02e95f1aSMarcin Radomski 
TEST(PcmTest,FormatToBits)46*02e95f1aSMarcin Radomski TEST(PcmTest, FormatToBits) {
47*02e95f1aSMarcin Radomski     // FIXME: Should we return 16 bits for INVALID?
48*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_INVALID), 16);
49*02e95f1aSMarcin Radomski 
50*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S16_LE), 16);
51*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S32_LE), 32);
52*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S8), 8);
53*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S24_LE), 32);
54*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S24_3LE), 24);
55*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S16_BE), 16);
56*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S24_BE), 32);
57*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S24_3BE), 24);
58*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_S32_BE), 32);
59*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_FLOAT_LE), 32);
60*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_format_to_bits(PCM_FORMAT_FLOAT_BE), 32);
61*02e95f1aSMarcin Radomski }
62*02e95f1aSMarcin Radomski 
TEST(PcmTest,OpenAndCloseOutPcm)63*02e95f1aSMarcin Radomski TEST(PcmTest, OpenAndCloseOutPcm) {
64*02e95f1aSMarcin Radomski     pcm *pcm_object = pcm_open(1000, 1000, PCM_OUT, &kDefaultConfig);
65*02e95f1aSMarcin Radomski     ASSERT_FALSE(pcm_is_ready(pcm_object));
66*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
67*02e95f1aSMarcin Radomski 
68*02e95f1aSMarcin Radomski     // assume card 0, device 0 is always available
69*02e95f1aSMarcin Radomski     pcm_object = pcm_open(0, 0, PCM_OUT, &kDefaultConfig);
70*02e95f1aSMarcin Radomski     ASSERT_TRUE(pcm_is_ready(pcm_object));
71*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
72*02e95f1aSMarcin Radomski 
73*02e95f1aSMarcin Radomski     pcm_object = pcm_open(0, 0, PCM_OUT | PCM_MMAP, &kDefaultConfig);
74*02e95f1aSMarcin Radomski     ASSERT_TRUE(pcm_is_ready(pcm_object));
75*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
76*02e95f1aSMarcin Radomski 
77*02e95f1aSMarcin Radomski     pcm_object = pcm_open(0, 0, PCM_OUT | PCM_MMAP | PCM_NOIRQ, &kDefaultConfig);
78*02e95f1aSMarcin Radomski     ASSERT_TRUE(pcm_is_ready(pcm_object));
79*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
80*02e95f1aSMarcin Radomski 
81*02e95f1aSMarcin Radomski     pcm_object = pcm_open(0, 0, PCM_OUT | PCM_NONBLOCK, &kDefaultConfig);
82*02e95f1aSMarcin Radomski     ASSERT_TRUE(pcm_is_ready(pcm_object));
83*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
84*02e95f1aSMarcin Radomski 
85*02e95f1aSMarcin Radomski     pcm_object = pcm_open(0, 0, PCM_OUT | PCM_MONOTONIC, &kDefaultConfig);
86*02e95f1aSMarcin Radomski     ASSERT_TRUE(pcm_is_ready(pcm_object));
87*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
88*02e95f1aSMarcin Radomski 
89*02e95f1aSMarcin Radomski     std::string name = "hw:0,0";
90*02e95f1aSMarcin Radomski     pcm_object = pcm_open_by_name(name.c_str(), PCM_OUT, &kDefaultConfig);
91*02e95f1aSMarcin Radomski     ASSERT_TRUE(pcm_is_ready(pcm_object));
92*02e95f1aSMarcin Radomski     ASSERT_EQ(pcm_close(pcm_object), 0);
93*02e95f1aSMarcin Radomski }
94*02e95f1aSMarcin Radomski 
TEST(PcmTest,OpenWithoutBlocking)95*02e95f1aSMarcin Radomski TEST(PcmTest, OpenWithoutBlocking) {
96*02e95f1aSMarcin Radomski     char loopback_device_info_path[120] = {};
97*02e95f1aSMarcin Radomski     snprintf(loopback_device_info_path, sizeof(loopback_device_info_path),
98*02e95f1aSMarcin Radomski             "/proc/asound/card%d/pcm%dp/info", kLoopbackCard, kLoopbackPlaybackDevice);
99*02e95f1aSMarcin Radomski 
100*02e95f1aSMarcin Radomski     std::ifstream info_file_stream{loopback_device_info_path};
101*02e95f1aSMarcin Radomski     if (!info_file_stream.is_open()) {
102*02e95f1aSMarcin Radomski         GTEST_SKIP();
103*02e95f1aSMarcin Radomski     }
104*02e95f1aSMarcin Radomski 
105*02e95f1aSMarcin Radomski     char buffer[256] = {};
106*02e95f1aSMarcin Radomski     int32_t subdevice_count = 0;
107*02e95f1aSMarcin Radomski     while (info_file_stream.good()) {
108*02e95f1aSMarcin Radomski         info_file_stream.getline(buffer, sizeof(buffer));
109*02e95f1aSMarcin Radomski         std::cout << buffer << std::endl;
110*02e95f1aSMarcin Radomski         std::string_view line{buffer};
111*02e95f1aSMarcin Radomski         if (line.find("subdevices_count") != std::string_view::npos) {
112*02e95f1aSMarcin Radomski             auto subdevice_count_string = line.substr(line.find(":") + 1);
113*02e95f1aSMarcin Radomski             std::cout << subdevice_count_string << std::endl;
114*02e95f1aSMarcin Radomski             subdevice_count = std::stoi(std::string{subdevice_count_string});
115*02e95f1aSMarcin Radomski         }
116*02e95f1aSMarcin Radomski     }
117*02e95f1aSMarcin Radomski 
118*02e95f1aSMarcin Radomski     ASSERT_GT(subdevice_count, 0);
119*02e95f1aSMarcin Radomski 
120*02e95f1aSMarcin Radomski     auto pcm_array = std::make_unique<pcm *[]>(subdevice_count);
121*02e95f1aSMarcin Radomski     std::thread *open_thread = new std::thread{[&pcm_array, subdevice_count] {
122*02e95f1aSMarcin Radomski         // Occupy all substreams
123*02e95f1aSMarcin Radomski         for (int32_t i = 0; i < subdevice_count; i++) {
124*02e95f1aSMarcin Radomski             pcm_array[i] = pcm_open(kLoopbackCard, kLoopbackPlaybackDevice, PCM_OUT,
125*02e95f1aSMarcin Radomski                     &kDefaultConfig);
126*02e95f1aSMarcin Radomski             EXPECT_TRUE(pcm_is_ready(pcm_array[i]));
127*02e95f1aSMarcin Radomski         }
128*02e95f1aSMarcin Radomski 
129*02e95f1aSMarcin Radomski         // Expect that pcm_open is not blocked in the kernel and return a bad_object pointer.
130*02e95f1aSMarcin Radomski         pcm *pcm_object = pcm_open(kLoopbackCard, kLoopbackPlaybackDevice, PCM_OUT,
131*02e95f1aSMarcin Radomski                     &kDefaultConfig);
132*02e95f1aSMarcin Radomski         if (pcm_is_ready(pcm_object)) {
133*02e95f1aSMarcin Radomski             // open_thread is blocked in kernel because of the substream is all occupied. pcm_open
134*02e95f1aSMarcin Radomski             // returns because the main thread has released all pcm structures in pcm_array. We just
135*02e95f1aSMarcin Radomski             // need to close the pcm_object here.
136*02e95f1aSMarcin Radomski             pcm_close(pcm_object);
137*02e95f1aSMarcin Radomski             return;
138*02e95f1aSMarcin Radomski         }
139*02e95f1aSMarcin Radomski 
140*02e95f1aSMarcin Radomski         // Release all substreams
141*02e95f1aSMarcin Radomski         for (int32_t i = 0; i < subdevice_count; i++) {
142*02e95f1aSMarcin Radomski             pcm_close(pcm_array[i]);
143*02e95f1aSMarcin Radomski             pcm_array[i] = nullptr;
144*02e95f1aSMarcin Radomski         }
145*02e95f1aSMarcin Radomski     }};
146*02e95f1aSMarcin Radomski 
147*02e95f1aSMarcin Radomski     static constexpr int64_t kTimeoutMs = 500;
148*02e95f1aSMarcin Radomski     std::this_thread::sleep_for(std::chrono::milliseconds(kTimeoutMs));
149*02e95f1aSMarcin Radomski     if (pcm_array[0] == nullptr) {
150*02e95f1aSMarcin Radomski         open_thread->join();
151*02e95f1aSMarcin Radomski     } else {
152*02e95f1aSMarcin Radomski         for (int32_t i = 0; i < subdevice_count; i++) {
153*02e95f1aSMarcin Radomski             pcm_close(pcm_array[i]);
154*02e95f1aSMarcin Radomski             pcm_array[i] = nullptr;
155*02e95f1aSMarcin Radomski         }
156*02e95f1aSMarcin Radomski         open_thread->join();
157*02e95f1aSMarcin Radomski         FAIL() << "The open_thread is blocked in kernel or the kTimeoutMs(" << kTimeoutMs <<
158*02e95f1aSMarcin Radomski                 ") is too short to complete";
159*02e95f1aSMarcin Radomski     }
160*02e95f1aSMarcin Radomski }
161*02e95f1aSMarcin Radomski 
162*02e95f1aSMarcin Radomski } // namespace testing
163*02e95f1aSMarcin Radomski } // namespace tinyalsa
164