summaryrefslogtreecommitdiff
path: root/src/audio_core/info_updater.cpp
diff options
context:
space:
mode:
authorGravatar bunnei2020-09-11 10:57:27 -0400
committerGravatar GitHub2020-09-11 10:57:27 -0400
commit324029d4f9fd2381f474e608a2859360324161e5 (patch)
treed2dc348235f05f20686c526f7092590f596f65c2 /src/audio_core/info_updater.cpp
parentMerge pull request #4597 from Morph1984/mjolnir-p2 (diff)
parentPreliminary effects (diff)
downloadyuzu-324029d4f9fd2381f474e608a2859360324161e5.tar.gz
yuzu-324029d4f9fd2381f474e608a2859360324161e5.tar.xz
yuzu-324029d4f9fd2381f474e608a2859360324161e5.zip
Merge pull request #4310 from ogniK5377/apollo-1-prod
audio_core: Apollo Part 1, AudioRenderer refactor
Diffstat (limited to 'src/audio_core/info_updater.cpp')
-rw-r--r--src/audio_core/info_updater.cpp517
1 files changed, 517 insertions, 0 deletions
diff --git a/src/audio_core/info_updater.cpp b/src/audio_core/info_updater.cpp
new file mode 100644
index 000000000..f53ce21a5
--- /dev/null
+++ b/src/audio_core/info_updater.cpp
@@ -0,0 +1,517 @@
1// Copyright 2020 yuzu Emulator Project
2// Licensed under GPLv2 or any later version
3// Refer to the license.txt file included.
4
5#include "audio_core/behavior_info.h"
6#include "audio_core/effect_context.h"
7#include "audio_core/info_updater.h"
8#include "audio_core/memory_pool.h"
9#include "audio_core/mix_context.h"
10#include "audio_core/sink_context.h"
11#include "audio_core/splitter_context.h"
12#include "audio_core/voice_context.h"
13#include "common/logging/log.h"
14
15namespace AudioCore {
16
17InfoUpdater::InfoUpdater(const std::vector<u8>& in_params, std::vector<u8>& out_params,
18 BehaviorInfo& behavior_info)
19 : in_params(in_params), out_params(out_params), behavior_info(behavior_info) {
20 ASSERT(
21 AudioCommon::CanConsumeBuffer(in_params.size(), 0, sizeof(AudioCommon::UpdateDataHeader)));
22 std::memcpy(&input_header, in_params.data(), sizeof(AudioCommon::UpdateDataHeader));
23 output_header.total_size = sizeof(AudioCommon::UpdateDataHeader);
24}
25
26InfoUpdater::~InfoUpdater() = default;
27
28bool InfoUpdater::UpdateBehaviorInfo(BehaviorInfo& in_behavior_info) {
29 if (input_header.size.behavior != sizeof(BehaviorInfo::InParams)) {
30 LOG_ERROR(Audio, "Behavior info is an invalid size, expecting 0x{:X} but got 0x{:X}",
31 sizeof(BehaviorInfo::InParams), input_header.size.behavior);
32 return false;
33 }
34
35 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset,
36 sizeof(BehaviorInfo::InParams))) {
37 LOG_ERROR(Audio, "Buffer is an invalid size!");
38 return false;
39 }
40
41 BehaviorInfo::InParams behavior_in{};
42 std::memcpy(&behavior_in, in_params.data() + input_offset, sizeof(BehaviorInfo::InParams));
43 input_offset += sizeof(BehaviorInfo::InParams);
44
45 // Make sure it's an audio revision we can actually support
46 if (!AudioCommon::IsValidRevision(behavior_in.revision)) {
47 LOG_ERROR(Audio, "Invalid input revision, revision=0x{:08X}", behavior_in.revision);
48 return false;
49 }
50
51 // Make sure that our behavior info revision matches the input
52 if (in_behavior_info.GetUserRevision() != behavior_in.revision) {
53 LOG_ERROR(Audio,
54 "User revision differs from input revision, expecting 0x{:08X} but got 0x{:08X}",
55 in_behavior_info.GetUserRevision(), behavior_in.revision);
56 return false;
57 }
58
59 // Update behavior info flags
60 in_behavior_info.ClearError();
61 in_behavior_info.UpdateFlags(behavior_in.flags);
62
63 return true;
64}
65
66bool InfoUpdater::UpdateMemoryPools(std::vector<ServerMemoryPoolInfo>& memory_pool_info) {
67 const auto force_mapping = behavior_info.IsMemoryPoolForceMappingEnabled();
68 const auto memory_pool_count = memory_pool_info.size();
69 const auto total_memory_pool_in = sizeof(ServerMemoryPoolInfo::InParams) * memory_pool_count;
70 const auto total_memory_pool_out = sizeof(ServerMemoryPoolInfo::OutParams) * memory_pool_count;
71
72 if (input_header.size.memory_pool != total_memory_pool_in) {
73 LOG_ERROR(Audio, "Memory pools are an invalid size, expecting 0x{:X} but got 0x{:X}",
74 total_memory_pool_in, input_header.size.memory_pool);
75 return false;
76 }
77
78 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset, total_memory_pool_in)) {
79 LOG_ERROR(Audio, "Buffer is an invalid size!");
80 return false;
81 }
82
83 std::vector<ServerMemoryPoolInfo::InParams> mempool_in(memory_pool_count);
84 std::vector<ServerMemoryPoolInfo::OutParams> mempool_out(memory_pool_count);
85
86 std::memcpy(mempool_in.data(), in_params.data() + input_offset, total_memory_pool_in);
87 input_offset += total_memory_pool_in;
88
89 // Update our memory pools
90 for (std::size_t i = 0; i < memory_pool_count; i++) {
91 if (!memory_pool_info[i].Update(mempool_in[i], mempool_out[i])) {
92 LOG_ERROR(Audio, "Failed to update memory pool {}!", i);
93 return false;
94 }
95 }
96
97 if (!AudioCommon::CanConsumeBuffer(out_params.size(), output_offset,
98 sizeof(BehaviorInfo::InParams))) {
99 LOG_ERROR(Audio, "Buffer is an invalid size!");
100 return false;
101 }
102
103 std::memcpy(out_params.data() + output_offset, mempool_out.data(), total_memory_pool_out);
104 output_offset += total_memory_pool_out;
105 output_header.size.memory_pool = static_cast<u32>(total_memory_pool_out);
106 return true;
107}
108
109bool InfoUpdater::UpdateVoiceChannelResources(VoiceContext& voice_context) {
110 const auto voice_count = voice_context.GetVoiceCount();
111 const auto voice_size = voice_count * sizeof(VoiceChannelResource::InParams);
112 std::vector<VoiceChannelResource::InParams> resources_in(voice_count);
113
114 if (input_header.size.voice_channel_resource != voice_size) {
115 LOG_ERROR(Audio, "VoiceChannelResource is an invalid size, expecting 0x{:X} but got 0x{:X}",
116 voice_size, input_header.size.voice_channel_resource);
117 return false;
118 }
119
120 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset, voice_size)) {
121 LOG_ERROR(Audio, "Buffer is an invalid size!");
122 return false;
123 }
124
125 std::memcpy(resources_in.data(), in_params.data() + input_offset, voice_size);
126 input_offset += voice_size;
127
128 // Update our channel resources
129 for (std::size_t i = 0; i < voice_count; i++) {
130 // Grab our channel resource
131 auto& resource = voice_context.GetChannelResource(i);
132 resource.Update(resources_in[i]);
133 }
134
135 return true;
136}
137
138bool InfoUpdater::UpdateVoices(VoiceContext& voice_context,
139 std::vector<ServerMemoryPoolInfo>& memory_pool_info,
140 VAddr audio_codec_dsp_addr) {
141 const auto voice_count = voice_context.GetVoiceCount();
142 std::vector<VoiceInfo::InParams> voice_in(voice_count);
143 std::vector<VoiceInfo::OutParams> voice_out(voice_count);
144
145 const auto voice_in_size = voice_count * sizeof(VoiceInfo::InParams);
146 const auto voice_out_size = voice_count * sizeof(VoiceInfo::OutParams);
147
148 if (input_header.size.voice != voice_in_size) {
149 LOG_ERROR(Audio, "Voices are an invalid size, expecting 0x{:X} but got 0x{:X}",
150 voice_in_size, input_header.size.voice);
151 return false;
152 }
153
154 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset, voice_in_size)) {
155 LOG_ERROR(Audio, "Buffer is an invalid size!");
156 return false;
157 }
158
159 std::memcpy(voice_in.data(), in_params.data() + input_offset, voice_in_size);
160 input_offset += voice_in_size;
161
162 // Set all voices to not be in use
163 for (std::size_t i = 0; i < voice_count; i++) {
164 voice_context.GetInfo(i).GetInParams().in_use = false;
165 }
166
167 // Update our voices
168 for (std::size_t i = 0; i < voice_count; i++) {
169 auto& in_params = voice_in[i];
170 const auto channel_count = static_cast<std::size_t>(in_params.channel_count);
171 // Skip if it's not currently in use
172 if (!in_params.is_in_use) {
173 continue;
174 }
175 // Voice states for each channel
176 std::array<VoiceState*, AudioCommon::MAX_CHANNEL_COUNT> voice_states{};
177 ASSERT(in_params.id < voice_count);
178
179 // Grab our current voice info
180 auto& voice_info = voice_context.GetInfo(static_cast<std::size_t>(in_params.id));
181
182 ASSERT(channel_count <= AudioCommon::MAX_CHANNEL_COUNT);
183
184 // Get all our channel voice states
185 for (std::size_t channel = 0; channel < channel_count; channel++) {
186 voice_states[channel] =
187 &voice_context.GetState(in_params.voice_channel_resource_ids[channel]);
188 }
189
190 if (in_params.is_new) {
191 // Default our values for our voice
192 voice_info.Initialize();
193 if (channel_count == 0 || channel_count > AudioCommon::MAX_CHANNEL_COUNT) {
194 continue;
195 }
196
197 // Zero out our voice states
198 for (std::size_t channel = 0; channel < channel_count; channel++) {
199 std::memset(voice_states[channel], 0, sizeof(VoiceState));
200 }
201 }
202
203 // Update our voice
204 voice_info.UpdateParameters(in_params, behavior_info);
205 // TODO(ogniK): Handle mapping errors with behavior info based on in params response
206
207 // Update our wave buffers
208 voice_info.UpdateWaveBuffers(in_params, voice_states, behavior_info);
209 voice_info.WriteOutStatus(voice_out[i], in_params, voice_states);
210 }
211
212 if (!AudioCommon::CanConsumeBuffer(out_params.size(), output_offset, voice_out_size)) {
213 LOG_ERROR(Audio, "Buffer is an invalid size!");
214 return false;
215 }
216 std::memcpy(out_params.data() + output_offset, voice_out.data(), voice_out_size);
217 output_offset += voice_out_size;
218 output_header.size.voice = static_cast<u32>(voice_out_size);
219 return true;
220}
221
222bool InfoUpdater::UpdateEffects(EffectContext& effect_context, bool is_active) {
223 const auto effect_count = effect_context.GetCount();
224 std::vector<EffectInfo::InParams> effect_in(effect_count);
225 std::vector<EffectInfo::OutParams> effect_out(effect_count);
226
227 const auto total_effect_in = effect_count * sizeof(EffectInfo::InParams);
228 const auto total_effect_out = effect_count * sizeof(EffectInfo::OutParams);
229
230 if (input_header.size.effect != total_effect_in) {
231 LOG_ERROR(Audio, "Effects are an invalid size, expecting 0x{:X} but got 0x{:X}",
232 total_effect_in, input_header.size.effect);
233 return false;
234 }
235
236 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset, total_effect_in)) {
237 LOG_ERROR(Audio, "Buffer is an invalid size!");
238 return false;
239 }
240
241 std::memcpy(effect_in.data(), in_params.data() + input_offset, total_effect_in);
242 input_offset += total_effect_in;
243
244 // Update effects
245 for (std::size_t i = 0; i < effect_count; i++) {
246 auto* info = effect_context.GetInfo(i);
247 if (effect_in[i].type != info->GetType()) {
248 info = effect_context.RetargetEffect(i, effect_in[i].type);
249 }
250
251 info->Update(effect_in[i]);
252
253 if ((!is_active && info->GetUsage() != UsageState::Initialized) ||
254 info->GetUsage() == UsageState::Stopped) {
255 effect_out[i].status = UsageStatus::Removed;
256 } else {
257 effect_out[i].status = UsageStatus::Used;
258 }
259 }
260
261 if (!AudioCommon::CanConsumeBuffer(out_params.size(), output_offset, total_effect_out)) {
262 LOG_ERROR(Audio, "Buffer is an invalid size!");
263 return false;
264 }
265
266 std::memcpy(out_params.data() + output_offset, effect_out.data(), total_effect_out);
267 output_offset += total_effect_out;
268 output_header.size.effect = static_cast<u32>(total_effect_out);
269
270 return true;
271}
272
273bool InfoUpdater::UpdateSplitterInfo(SplitterContext& splitter_context) {
274 std::size_t start_offset = input_offset;
275 std::size_t bytes_read{};
276 // Update splitter context
277 if (!splitter_context.Update(in_params, input_offset, bytes_read)) {
278 LOG_ERROR(Audio, "Failed to update splitter context!");
279 return false;
280 }
281
282 const auto consumed = input_offset - start_offset;
283
284 if (input_header.size.splitter != consumed) {
285 LOG_ERROR(Audio, "Splitters is an invalid size, expecting 0x{:X} but got 0x{:X}",
286 bytes_read, input_header.size.splitter);
287 return false;
288 }
289
290 return true;
291}
292
293ResultCode InfoUpdater::UpdateMixes(MixContext& mix_context, std::size_t mix_buffer_count,
294 SplitterContext& splitter_context,
295 EffectContext& effect_context) {
296 std::vector<MixInfo::InParams> mix_in_params;
297
298 if (!behavior_info.IsMixInParameterDirtyOnlyUpdateSupported()) {
299 // If we're not dirty, get ALL mix in parameters
300 const auto context_mix_count = mix_context.GetCount();
301 const auto total_mix_in = context_mix_count * sizeof(MixInfo::InParams);
302 if (input_header.size.mixer != total_mix_in) {
303 LOG_ERROR(Audio, "Mixer is an invalid size, expecting 0x{:X} but got 0x{:X}",
304 total_mix_in, input_header.size.mixer);
305 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
306 }
307
308 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset, total_mix_in)) {
309 LOG_ERROR(Audio, "Buffer is an invalid size!");
310 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
311 }
312
313 mix_in_params.resize(context_mix_count);
314 std::memcpy(mix_in_params.data(), in_params.data() + input_offset, total_mix_in);
315
316 input_offset += total_mix_in;
317 } else {
318 // Only update the "dirty" mixes
319 MixInfo::DirtyHeader dirty_header{};
320 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset,
321 sizeof(MixInfo::DirtyHeader))) {
322 LOG_ERROR(Audio, "Buffer is an invalid size!");
323 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
324 }
325
326 std::memcpy(&dirty_header, in_params.data() + input_offset, sizeof(MixInfo::DirtyHeader));
327 input_offset += sizeof(MixInfo::DirtyHeader);
328
329 const auto total_mix_in =
330 dirty_header.mixer_count * sizeof(MixInfo::InParams) + sizeof(MixInfo::DirtyHeader);
331
332 if (input_header.size.mixer != total_mix_in) {
333 LOG_ERROR(Audio, "Mixer is an invalid size, expecting 0x{:X} but got 0x{:X}",
334 total_mix_in, input_header.size.mixer);
335 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
336 }
337
338 if (dirty_header.mixer_count != 0) {
339 mix_in_params.resize(dirty_header.mixer_count);
340 std::memcpy(mix_in_params.data(), in_params.data() + input_offset,
341 mix_in_params.size() * sizeof(MixInfo::InParams));
342 input_offset += mix_in_params.size() * sizeof(MixInfo::InParams);
343 }
344 }
345
346 // Get our total input count
347 const auto mix_count = mix_in_params.size();
348
349 if (!behavior_info.IsMixInParameterDirtyOnlyUpdateSupported()) {
350 // Only verify our buffer count if we're not dirty
351 std::size_t total_buffer_count{};
352 for (std::size_t i = 0; i < mix_count; i++) {
353 const auto& in = mix_in_params[i];
354 total_buffer_count += in.buffer_count;
355 if (in.dest_mix_id > mix_count && in.dest_mix_id != AudioCommon::NO_MIX &&
356 in.mix_id != AudioCommon::FINAL_MIX) {
357 LOG_ERROR(
358 Audio,
359 "Invalid mix destination, mix_id={:X}, dest_mix_id={:X}, mix_buffer_count={:X}",
360 in.mix_id, in.dest_mix_id, mix_buffer_count);
361 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
362 }
363 }
364
365 if (total_buffer_count > mix_buffer_count) {
366 LOG_ERROR(Audio,
367 "Too many mix buffers used! mix_buffer_count={:X}, requesting_buffers={:X}",
368 mix_buffer_count, total_buffer_count);
369 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
370 }
371 }
372
373 if (mix_buffer_count == 0) {
374 LOG_ERROR(Audio, "No mix buffers!");
375 return AudioCommon::Audren::ERR_INVALID_PARAMETERS;
376 }
377
378 bool should_sort = false;
379 for (std::size_t i = 0; i < mix_count; i++) {
380 const auto& mix_in = mix_in_params[i];
381 std::size_t target_mix{};
382 if (behavior_info.IsMixInParameterDirtyOnlyUpdateSupported()) {
383 target_mix = mix_in.mix_id;
384 } else {
385 // Non dirty supported games just use i instead of the actual mix_id
386 target_mix = i;
387 }
388 auto& mix_info = mix_context.GetInfo(target_mix);
389 auto& mix_info_params = mix_info.GetInParams();
390 if (mix_info_params.in_use != mix_in.in_use) {
391 mix_info_params.in_use = mix_in.in_use;
392 mix_info.ResetEffectProcessingOrder();
393 should_sort = true;
394 }
395
396 if (mix_in.in_use) {
397 should_sort |= mix_info.Update(mix_context.GetEdgeMatrix(), mix_in, behavior_info,
398 splitter_context, effect_context);
399 }
400 }
401
402 if (should_sort && behavior_info.IsSplitterSupported()) {
403 // Sort our splitter data
404 if (!mix_context.TsortInfo(splitter_context)) {
405 return AudioCommon::Audren::ERR_SPLITTER_SORT_FAILED;
406 }
407 }
408
409 // TODO(ogniK): Sort when splitter is suppoorted
410
411 return RESULT_SUCCESS;
412}
413
414bool InfoUpdater::UpdateSinks(SinkContext& sink_context) {
415 const auto sink_count = sink_context.GetCount();
416 std::vector<SinkInfo::InParams> sink_in_params(sink_count);
417 const auto total_sink_in = sink_count * sizeof(SinkInfo::InParams);
418
419 if (input_header.size.sink != total_sink_in) {
420 LOG_ERROR(Audio, "Sinks are an invalid size, expecting 0x{:X} but got 0x{:X}",
421 total_sink_in, input_header.size.effect);
422 return false;
423 }
424
425 if (!AudioCommon::CanConsumeBuffer(in_params.size(), input_offset, total_sink_in)) {
426 LOG_ERROR(Audio, "Buffer is an invalid size!");
427 return false;
428 }
429
430 std::memcpy(sink_in_params.data(), in_params.data() + input_offset, total_sink_in);
431 input_offset += total_sink_in;
432
433 // TODO(ogniK): Properly update sinks
434 if (!sink_in_params.empty()) {
435 sink_context.UpdateMainSink(sink_in_params[0]);
436 }
437
438 output_header.size.sink = static_cast<u32>(0x20 * sink_count);
439 output_offset += 0x20 * sink_count;
440 return true;
441}
442
443bool InfoUpdater::UpdatePerformanceBuffer() {
444 output_header.size.performance = 0x10;
445 output_offset += 0x10;
446 return true;
447}
448
449bool InfoUpdater::UpdateErrorInfo(BehaviorInfo& in_behavior_info) {
450 const auto total_beahvior_info_out = sizeof(BehaviorInfo::OutParams);
451
452 if (!AudioCommon::CanConsumeBuffer(out_params.size(), output_offset, total_beahvior_info_out)) {
453 LOG_ERROR(Audio, "Buffer is an invalid size!");
454 return false;
455 }
456
457 BehaviorInfo::OutParams behavior_info_out{};
458 behavior_info.CopyErrorInfo(behavior_info_out);
459
460 std::memcpy(out_params.data() + output_offset, &behavior_info_out, total_beahvior_info_out);
461 output_offset += total_beahvior_info_out;
462 output_header.size.behavior = total_beahvior_info_out;
463
464 return true;
465}
466
467struct RendererInfo {
468 u64_le elasped_frame_count{};
469 INSERT_PADDING_WORDS(2);
470};
471static_assert(sizeof(RendererInfo) == 0x10, "RendererInfo is an invalid size");
472
473bool InfoUpdater::UpdateRendererInfo(std::size_t elapsed_frame_count) {
474 const auto total_renderer_info_out = sizeof(RendererInfo);
475 if (!AudioCommon::CanConsumeBuffer(out_params.size(), output_offset, total_renderer_info_out)) {
476 LOG_ERROR(Audio, "Buffer is an invalid size!");
477 return false;
478 }
479 RendererInfo out{};
480 out.elasped_frame_count = elapsed_frame_count;
481 std::memcpy(out_params.data() + output_offset, &out, total_renderer_info_out);
482 output_offset += total_renderer_info_out;
483 output_header.size.render_info = total_renderer_info_out;
484
485 return true;
486}
487
488bool InfoUpdater::CheckConsumedSize() const {
489 if (output_offset != out_params.size()) {
490 LOG_ERROR(Audio, "Output is not consumed! Consumed {}, but requires {}. {} bytes remaining",
491 output_offset, out_params.size(), out_params.size() - output_offset);
492 return false;
493 }
494 /*if (input_offset != in_params.size()) {
495 LOG_ERROR(Audio, "Input is not consumed!");
496 return false;
497 }*/
498 return true;
499}
500
501bool InfoUpdater::WriteOutputHeader() {
502 if (!AudioCommon::CanConsumeBuffer(out_params.size(), 0,
503 sizeof(AudioCommon::UpdateDataHeader))) {
504 LOG_ERROR(Audio, "Buffer is an invalid size!");
505 return false;
506 }
507 output_header.revision = AudioCommon::CURRENT_PROCESS_REVISION;
508 const auto& sz = output_header.size;
509 output_header.total_size += sz.behavior + sz.memory_pool + sz.voice +
510 sz.voice_channel_resource + sz.effect + sz.mixer + sz.sink +
511 sz.performance + sz.splitter + sz.render_info;
512
513 std::memcpy(out_params.data(), &output_header, sizeof(AudioCommon::UpdateDataHeader));
514 return true;
515}
516
517} // namespace AudioCore