1 /* 2 * Copyright 1991-1999, Be Incorporated. 3 * Copyright (c) 1999-2000, Eric Moon. 4 * All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions 8 * are met: 9 * 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions, and the following disclaimer. 12 * 13 * 2. Redistributions in binary form must reproduce the above copyright 14 * notice, this list of conditions, and the following disclaimer in the 15 * documentation and/or other materials provided with the distribution. 16 * 17 * 3. The name of the author may not be used to endorse or promote products 18 * derived from this software without specific prior written permission. 19 * 20 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR 21 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 22 * OF TITLE, NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS FOR A PARTICULAR 23 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY 24 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 25 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 26 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 27 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR 28 * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 29 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 30 */ 31 32 33 // LoggingConsumer.cpp 34 35 #include "LoggingConsumer.h" 36 #include "LogWriter.h" 37 #include <media/MediaRoster.h> 38 #include <media/TimeSource.h> 39 #include <media/ParameterWeb.h> 40 #include <media/Buffer.h> 41 #include <kernel/OS.h> 42 #include <stdio.h> 43 #include <string.h> 44 45 // e.moon [11jun99] 46 #include <Debug.h> 47 48 // id's of the node's BParameters 49 const int32 INPUT_NULL_PARAM = 1; 50 const int32 LATENCY_PARAM = 2; 51 const int32 OUTPUT_NULL_PARAM = 3; 52 const int32 CPU_NULL_PARAM = 11; 53 const int32 CPU_SPIN_PARAM = 12; 54 const int32 PRIO_NULL_PARAM = 21; 55 const int32 PRIORITY_PARAM = 22; 56 57 // build the LoggingConsumer's BParameterWeb 58 static BParameterWeb* build_parameter_web() 59 { 60 BParameterWeb* web = new BParameterWeb; 61 62 BParameterGroup* mainGroup = web->MakeGroup("LoggingConsumer Parameters"); 63 BParameterGroup* group = mainGroup->MakeGroup("Latency control"); 64 BParameter* nullParam = group->MakeNullParameter(INPUT_NULL_PARAM, B_MEDIA_NO_TYPE, "Latency", B_GENERIC); 65 BParameter* latencyParam = group->MakeContinuousParameter(LATENCY_PARAM, B_MEDIA_NO_TYPE, "", 66 B_GAIN, "ms", 5, 100, 5); 67 nullParam->AddOutput(latencyParam); 68 latencyParam->AddInput(nullParam); 69 70 group = mainGroup->MakeGroup("CPU percentage"); 71 nullParam = group->MakeNullParameter(CPU_NULL_PARAM, B_MEDIA_NO_TYPE, "CPU spin percentage", B_GENERIC); 72 BContinuousParameter* cpuParam = group->MakeContinuousParameter(CPU_SPIN_PARAM, B_MEDIA_NO_TYPE, "", 73 B_GAIN, "percent", 5, 80, 5); 74 nullParam->AddOutput(cpuParam); 75 cpuParam->AddInput(nullParam); 76 77 group = mainGroup->MakeGroup("Priority"); 78 nullParam = group->MakeNullParameter(PRIO_NULL_PARAM, B_MEDIA_NO_TYPE, "Thread priority", B_GENERIC); 79 BDiscreteParameter* prioParam = group->MakeDiscreteParameter(PRIORITY_PARAM, B_MEDIA_NO_TYPE, "", B_GENERIC); 80 prioParam->AddItem(5, "B_LOW_PRIORITY"); 81 prioParam->AddItem(10, "B_NORMAL_PRIORITY"); 82 prioParam->AddItem(15, "B_DISPLAY_PRIORITY"); 83 prioParam->AddItem(20, "B_URGENT_DISPLAY_PRIORITY"); 84 prioParam->AddItem(100, "B_REAL_TIME_DISPLAY_PRIORITY"); 85 prioParam->AddItem(110, "B_URGENT_PRIORITY"); 86 prioParam->AddItem(120, "B_REAL_TIME_PRIORITY"); 87 88 return web; 89 } 90 91 // -------------------- 92 // LoggingConsumer class implementation 93 LoggingConsumer::LoggingConsumer( 94 const entry_ref& logFile, 95 BMediaAddOn* pAddOn) 96 97 : BMediaNode("LoggingConsumer"), 98 BBufferConsumer(B_MEDIA_UNKNOWN_TYPE), 99 BControllable(), 100 BMediaEventLooper(), 101 mLogRef(logFile), 102 mWeb(NULL), 103 mLateBuffers(0), 104 mLatency(50 * 1000), // default to 50 milliseconds 105 mSpinPercentage(0.10), // default to spinning 10% of total latency 106 mPriority(B_URGENT_DISPLAY_PRIORITY), // !!! testing; will be B_REAL_TIME_PRIORITY for release 107 mLastLatencyChange(0), 108 mLastSpinChange(0), 109 mLastPrioChange(0), 110 m_pAddOn(pAddOn) 111 { 112 // spin off the logging thread 113 mLogger = new LogWriter(logFile); 114 115 // parameter-web init moved to NodeRegistered() 116 // e.moon [11jun99] 117 } 118 119 LoggingConsumer::~LoggingConsumer() 120 { 121 PRINT(("~LoggingConsumer()\n")); 122 BMediaEventLooper::Quit(); 123 // ahem: 124 // "Once you've called BControllable::SetParameterWeb(), the node takes 125 // responsibility for the parameter web object and you shouldn't delete it. " 126 // SetParameterWeb(NULL); 127 // delete mWeb; 128 129 // delete the logging thread only after the looper thread has quit, otherwise there's 130 // a potential race condition with the looper thread trying to write to the now- 131 // deleted log 132 delete mLogger; 133 } 134 135 // 136 // Log message filtering control 137 // 138 139 void 140 LoggingConsumer::SetEnabled(log_what what, bool enable) 141 { 142 mLogger->SetEnabled(what, enable); 143 } 144 145 void 146 LoggingConsumer::EnableAllMessages() 147 { 148 mLogger->EnableAllMessages(); 149 } 150 151 void 152 LoggingConsumer::DisableAllMessages() 153 { 154 mLogger->DisableAllMessages(); 155 } 156 157 // 158 // BMediaNode methods 159 // 160 161 162 BMediaAddOn* 163 LoggingConsumer::AddOn(int32 *internal_id) const 164 { 165 PRINT(("~LoggingConsumer::AddOn()\n")); 166 // e.moon [11jun99] 167 if(m_pAddOn) { 168 *internal_id = 0; 169 return m_pAddOn; 170 } else 171 return NULL; 172 } 173 174 void 175 LoggingConsumer::SetRunMode(run_mode mode) 176 { 177 // !!! Need to handle offline mode etc. properly! 178 log_message logMsg; 179 logMsg.now = TimeSource()->Now(); 180 mLogger->Log(LOG_SET_RUN_MODE, logMsg); 181 182 BMediaEventLooper::SetRunMode(mode); 183 } 184 185 void 186 LoggingConsumer::Preroll() 187 { 188 log_message logMsg; 189 logMsg.now = TimeSource()->Now(); 190 mLogger->Log(LOG_PREROLL, logMsg); 191 192 BMediaEventLooper::Preroll(); 193 } 194 195 void 196 LoggingConsumer::SetTimeSource(BTimeSource* time_source) 197 { 198 log_message logMsg; 199 logMsg.now = TimeSource()->Now(); 200 mLogger->Log(LOG_SET_TIME_SOURCE, logMsg); 201 202 BMediaNode::SetTimeSource(time_source); 203 } 204 205 status_t 206 LoggingConsumer::RequestCompleted(const media_request_info &info) 207 { 208 log_message logMsg; 209 logMsg.now = TimeSource()->Now(); 210 mLogger->Log(LOG_REQUEST_COMPLETED, logMsg); 211 212 return BMediaNode::RequestCompleted(info); 213 } 214 215 // e.moon [11jun99; testing add-on] 216 status_t 217 LoggingConsumer::DeleteHook(BMediaNode* pNode) { 218 PRINT(("LoggingConsumer::DeleteHook(%p)\n", pNode)); 219 return BBufferConsumer::DeleteHook(pNode); 220 // ASSERT(pNode == this); 221 // delete this; 222 // return B_OK; 223 } 224 225 // 226 // BControllable methods 227 // 228 229 status_t 230 LoggingConsumer::GetParameterValue(int32 id, bigtime_t* last_change, void* value, size_t* ioSize) 231 { 232 log_message logMsg; 233 logMsg.now = TimeSource()->Now(); 234 logMsg.param.id = id; 235 mLogger->Log(LOG_GET_PARAM_VALUE, logMsg); 236 237 // return an error if the caller hasn't reserved enough space for the parameter data. 238 // we know that all of our parameters fit in a float or int32 (4 bytes), so we can just 239 // check for it once here, instead of on a per-parameter basis 240 if (*ioSize < sizeof(float)) return B_ERROR; 241 242 // write out the designated parameter data 243 switch (id) 244 { 245 case LATENCY_PARAM: 246 *last_change = mLastLatencyChange; 247 *((float*) value) = mLatency / 1000; // the BParameter reads milliseconds, not microseconds 248 *ioSize = sizeof(float); 249 break; 250 251 case CPU_SPIN_PARAM: 252 *last_change = mLastSpinChange; 253 *((float*) value) = mSpinPercentage; 254 *ioSize = sizeof(float); 255 break; 256 257 case PRIORITY_PARAM: 258 *last_change = mLastPrioChange; 259 *((int32*) value) = mPriority; 260 *ioSize = sizeof(int32); 261 break; 262 263 default: 264 return B_ERROR; 265 } 266 267 return B_OK; 268 } 269 270 void 271 LoggingConsumer::SetParameterValue(int32 id, bigtime_t performance_time, const void* value, size_t size) 272 { 273 log_message logMsg; 274 logMsg.now = TimeSource()->Now(); 275 logMsg.param.id = id; 276 mLogger->Log(LOG_SET_PARAM_VALUE, logMsg); 277 278 // if it's one of our parameters, enqueue a "set parameter" event for handling at the appropriate time 279 switch (id) 280 { 281 case LATENCY_PARAM: 282 case CPU_SPIN_PARAM: 283 case PRIORITY_PARAM: 284 { 285 // !!! Change from B_USER_EVENT to B_SET_PARAMETER once it's defined 286 media_timed_event event(performance_time, BTimedEventQueue::B_USER_EVENT, 287 (void*) value, BTimedEventQueue::B_NO_CLEANUP, size, id, NULL); 288 EventQueue()->AddEvent(event); 289 } 290 break; 291 292 default: // do nothing for other parameter IDs 293 break; 294 } 295 return; 296 } 297 298 // 299 // BBufferConsumer methods 300 // 301 302 status_t 303 LoggingConsumer::HandleMessage(int32 message, const void *data, size_t size) 304 { 305 log_message logMsg; 306 logMsg.now = TimeSource()->Now(); 307 mLogger->Log(LOG_HANDLE_MESSAGE, logMsg); 308 309 // try each of our superclasses to handle the message 310 status_t err; 311 err = BControllable::HandleMessage(message, data, size); 312 if (err) err = BBufferConsumer::HandleMessage(message, data, size); 313 if (err) err = BMediaNode::HandleMessage(message, data, size); 314 return err; 315 } 316 317 // all of these next methods are pure virtual in BBufferConsumer 318 319 status_t 320 LoggingConsumer::AcceptFormat(const media_destination& dest, media_format* format) 321 { 322 char formatStr[256]; 323 string_for_format(*format, formatStr, 255); 324 PRINT(("LoggingConsumer::AcceptFormat:\n\tformat %s\n", formatStr)); 325 326 log_message logMsg; 327 logMsg.now = TimeSource()->Now(); 328 mLogger->Log(LOG_ACCEPT_FORMAT, logMsg); 329 330 // return an error if this isn't really our one input's destination 331 if (dest != mInput.destination) return B_MEDIA_BAD_DESTINATION; 332 333 // the destination given really is our input, and we accept any kind of media data, 334 // so now we just confirm that we can handle whatever the producer asked for. 335 return B_OK; 336 } 337 338 status_t 339 LoggingConsumer::GetNextInput(int32* cookie, media_input* out_input) 340 { 341 // we have a single hardcoded input that can accept any kind of media data 342 if (0 == *cookie) 343 { 344 mInput.format.type = B_MEDIA_UNKNOWN_TYPE; // accept any format 345 346 *out_input = mInput; 347 *cookie = 1; 348 return B_OK; 349 } 350 else return B_BAD_INDEX; 351 } 352 353 void 354 LoggingConsumer::DisposeInputCookie(int32 /*cookie*/ ) 355 { 356 // we don't use any kind of state or extra storage for iterating over our 357 // inputs, so we don't have to do any special disposal of input cookies. 358 } 359 360 void 361 LoggingConsumer::BufferReceived(BBuffer* buffer) 362 { 363 bigtime_t bufferStart = buffer->Header()->start_time; 364 bigtime_t now = TimeSource()->Now(); 365 bigtime_t how_early = bufferStart - EventLatency() - SchedulingLatency() - now; 366 367 log_message logMsg; 368 logMsg.now = now; 369 logMsg.buffer_data.start_time = bufferStart; 370 logMsg.buffer_data.offset = how_early; 371 mLogger->Log(LOG_BUFFER_RECEIVED, logMsg); 372 373 // There's a special case here with handling B_MEDIA_PARAMETERS buffers. 374 // These contain sets of parameter value changes, with their own performance 375 // times embedded in the buffers. So, we want to dispatch those parameter 376 // changes as their own events rather than pushing this buffer on the queue to 377 // be handled later. 378 if (B_MEDIA_PARAMETERS == buffer->Header()->type) 379 { 380 ApplyParameterData(buffer->Data(), buffer->SizeUsed()); 381 buffer->Recycle(); 382 } 383 else // ahh, it's a regular media buffer, so push it on the event queue 384 { 385 status_t err; 386 media_timed_event event(buffer->Header()->start_time, BTimedEventQueue::B_HANDLE_BUFFER, 387 buffer, BTimedEventQueue::B_RECYCLE_BUFFER); 388 err = EventQueue()->AddEvent(event); 389 390 // HandleEvent() will recycle the buffer. However, if we incurred an error trying to 391 // put the event into the queue, we have to recycle it ourselves, since HandleEvent() 392 // will never see the buffer in that case. 393 if (err) buffer->Recycle(); 394 } 395 } 396 397 void 398 LoggingConsumer::ProducerDataStatus(const media_destination& for_whom, int32 status, bigtime_t at_performance_time) 399 { 400 log_message logMsg; 401 logMsg.now = TimeSource()->Now(); 402 logMsg.data_status.status = status; 403 mLogger->Log(LOG_PRODUCER_DATA_STATUS, logMsg); 404 405 if (for_whom == mInput.destination) 406 { 407 media_timed_event event(at_performance_time, BTimedEventQueue::B_DATA_STATUS, 408 &mInput, BTimedEventQueue::B_NO_CLEANUP, status, 0, NULL); 409 EventQueue()->AddEvent(event); 410 } 411 } 412 413 status_t 414 LoggingConsumer::GetLatencyFor(const media_destination& for_whom, bigtime_t* out_latency, media_node_id* out_timesource) 415 { 416 // make sure this is one of my valid inputs 417 if (for_whom != mInput.destination) return B_MEDIA_BAD_DESTINATION; 418 419 // report internal latency + downstream latency here, NOT including scheduling latency. 420 // we're a final consumer (no outputs), so we have no downstream latency. 421 *out_latency = mLatency; 422 *out_timesource = TimeSource()->ID(); 423 return B_OK; 424 } 425 426 status_t 427 LoggingConsumer::Connected( 428 const media_source& producer, 429 const media_destination& where, 430 const media_format& with_format, 431 media_input* out_input) 432 { 433 434 char formatStr[256]; 435 string_for_format(with_format, formatStr, 255); 436 PRINT(("LoggingConsumer::Connected:\n\tformat %s\n", formatStr)); 437 string_for_format(mInput.format, formatStr, 255); 438 PRINT(("\tinput format %s\n", formatStr)); 439 440 log_message logMsg; 441 logMsg.now = TimeSource()->Now(); 442 mLogger->Log(LOG_CONNECTED, logMsg); 443 444 if (where != mInput.destination) return B_MEDIA_BAD_DESTINATION; 445 446 // calculate my latency here, because it may depend on buffer sizes/durations, then 447 // tell the BMediaEventLooper how early we need to get the buffers 448 SetEventLatency(mLatency); 449 450 // record useful information about the connection, and return success 451 // * e.moon [14jun99]: stores format 452 mInput.format = with_format; 453 mInput.source = producer; 454 *out_input = mInput; 455 return B_OK; 456 } 457 458 void 459 LoggingConsumer::Disconnected( 460 const media_source& producer, 461 const media_destination& where) 462 { 463 log_message logMsg; 464 logMsg.now = TimeSource()->Now(); 465 mLogger->Log(LOG_DISCONNECTED, logMsg); 466 467 // wipe out our input record 468 memset(&mInput, 0, sizeof(mInput)); 469 } 470 471 status_t 472 LoggingConsumer::FormatChanged( 473 const media_source& producer, 474 const media_destination& consumer, 475 int32 change_tag, 476 const media_format& format) 477 { 478 log_message logMsg; 479 logMsg.now = TimeSource()->Now(); 480 mLogger->Log(LOG_FORMAT_CHANGED, logMsg); 481 482 return B_OK; 483 } 484 485 status_t 486 LoggingConsumer::SeekTagRequested( 487 const media_destination& destination, 488 bigtime_t in_target_time, 489 uint32 in_flags, 490 media_seek_tag* out_seek_tag, 491 bigtime_t* out_tagged_time, 492 uint32* out_flags) 493 { 494 log_message logMsg; 495 logMsg.now = TimeSource()->Now(); 496 mLogger->Log(LOG_SEEK_TAG, logMsg); 497 498 return B_OK; 499 } 500 501 // 502 // BMediaEventLooper virtual methods 503 // 504 505 void 506 LoggingConsumer::NodeRegistered() 507 { 508 log_message logMsg; 509 logMsg.now = TimeSource()->Now(); 510 mLogger->Log(LOG_REGISTERED, logMsg); 511 512 // publish our parameter web 513 mWeb = build_parameter_web(); 514 SetParameterWeb(mWeb); 515 516 // Set our priority and start the BMediaEventLooper's thread 517 SetPriority(mPriority); 518 Run(); 519 520 // Initialize as much of our input as we can, now that the Media Kit really "knows" about us 521 mInput.destination.port = ControlPort(); 522 mInput.destination.id = 0; 523 mInput.node = Node(); 524 strcpy(mInput.name, "Logged input"); 525 } 526 527 void 528 LoggingConsumer::Start(bigtime_t performance_time) 529 { 530 PRINT(("LoggingConsumer::Start(%Ld): now %Ld\n", performance_time, TimeSource()->Now())); 531 532 log_message logMsg; 533 logMsg.now = TimeSource()->Now(); 534 mLogger->Log(LOG_START, logMsg); 535 536 BMediaEventLooper::Start(performance_time); 537 } 538 539 void 540 LoggingConsumer::Stop(bigtime_t performance_time, bool immediate) 541 { 542 log_message logMsg; 543 logMsg.now = TimeSource()->Now(); 544 mLogger->Log(LOG_STOP, logMsg); 545 546 BMediaEventLooper::Stop(performance_time, immediate); 547 } 548 549 void 550 LoggingConsumer::Seek(bigtime_t media_time, bigtime_t performance_time) 551 { 552 log_message logMsg; 553 logMsg.now = TimeSource()->Now(); 554 mLogger->Log(LOG_SEEK, logMsg); 555 556 BMediaEventLooper::Seek(media_time, performance_time); 557 } 558 559 void 560 LoggingConsumer::TimeWarp(bigtime_t at_real_time, bigtime_t to_performance_time) 561 { 562 log_message logMsg; 563 logMsg.now = TimeSource()->Now(); 564 mLogger->Log(LOG_TIMEWARP, logMsg); 565 566 BMediaEventLooper::TimeWarp(at_real_time, to_performance_time); 567 } 568 569 void 570 LoggingConsumer::HandleEvent(const media_timed_event *event, bigtime_t /* lateness */, bool /* realTimeEvent */) 571 { 572 log_message logMsg; 573 logMsg.now = TimeSource()->Now(); 574 mLogger->Log(LOG_HANDLE_EVENT, logMsg); 575 576 switch (event->type) 577 { 578 case BTimedEventQueue::B_HANDLE_BUFFER: 579 { 580 BBuffer* buffer = const_cast<BBuffer*>((BBuffer*) event->pointer); 581 if (buffer) 582 { 583 media_header* hdr = buffer->Header(); 584 if (hdr->destination == mInput.destination.id) 585 { 586 bigtime_t now = TimeSource()->Now(); 587 bigtime_t perf_time = hdr->start_time; 588 589 // the how_early calculated here doesn't include scheduling latency because 590 // we've already been scheduled to handle the buffer 591 bigtime_t how_early = perf_time - mLatency - now; 592 593 // logMsg.now is already set 594 logMsg.buffer_data.start_time = perf_time; 595 logMsg.buffer_data.offset = how_early; 596 mLogger->Log(LOG_BUFFER_HANDLED, logMsg); 597 598 // if the buffer is late, we ignore it and report the fact to the producer 599 // who sent it to us 600 if (how_early < 0) 601 { 602 mLateBuffers++; 603 NotifyLateProducer(mInput.source, -how_early, perf_time); 604 } 605 else 606 { 607 // burn some percentage of our stated latency in CPU time (controlled by 608 // a BParameter). this simulates a user-configurable amount of CPU cost 609 // associated with the consumer. 610 bigtime_t spin_start = ::system_time(); 611 bigtime_t spin_now = spin_start; 612 bigtime_t usecToSpin = bigtime_t(mSpinPercentage / 100.0 * mLatency); 613 while (spin_now - spin_start < usecToSpin) 614 { 615 for (long k = 0; k < 1000000; k++) { /* intentionally blank */ } 616 spin_now = ::system_time(); 617 } 618 } 619 620 // we're done "processing the buffer;" now we recycle it and return to the loop 621 buffer->Recycle(); 622 } 623 else 624 { 625 //fprintf(stderr, "* Woah! Got a buffer for a different destination!\n"); 626 } 627 } 628 } 629 break; 630 631 // !!! change to B_PARAMETER as soon as it's available 632 633 // +++++ e.moon [16jun99] 634 // !!! this can't be right: the parameter value is accessed by the pointer 635 // originally passed to SetParameterValue(). there's no guarantee that 636 // value's still valid, is there? 637 638 case BTimedEventQueue::B_USER_EVENT: 639 { 640 size_t dataSize = size_t(event->data); 641 int32 param = int32(event->bigdata); 642 logMsg.param.id = param; 643 644 // handle the message if there's sufficient data provided. we only check against 645 // sizeof(float) because all of our parameters happen to be 4 bytes. if various 646 // parameters took different amounts of data, we'd check the size on a per-parameter 647 // basis. 648 if (dataSize >= sizeof(float)) switch (param) 649 { 650 case LATENCY_PARAM: 651 { 652 float value = *((float*) event->pointer); 653 mLatency = bigtime_t(value* 1000); 654 mLastLatencyChange = logMsg.now; 655 656 // my latency just changed, so reconfigure the BMediaEventLooper 657 // to give me my events at the proper time 658 SetEventLatency(mLatency); 659 660 // tell the producer that my latency changed, and broadcast a message 661 // about the parameter change to any applications that may be looking 662 // for it through the BMediaRoster::StartWatching() mechanism. 663 // 664 // if we had more than one input, we'd need to tell *all* producers about 665 // the change in our latency. 666 SendLatencyChange(mInput.source, mInput.destination, EventLatency() + SchedulingLatency()); 667 BroadcastNewParameterValue(logMsg.now, param, &value, sizeof(value)); 668 669 // log the new latency value, for recordkeeping 670 logMsg.param.value = value; 671 mLogger->Log(LOG_SET_PARAM_HANDLED, logMsg); 672 } 673 break; 674 675 case CPU_SPIN_PARAM: 676 { 677 float value = *((float*) event->pointer); 678 mSpinPercentage = value; 679 mLastSpinChange = logMsg.now; 680 BroadcastNewParameterValue(logMsg.now, param, &value, sizeof(value)); 681 logMsg.param.value = value; 682 mLogger->Log(LOG_SET_PARAM_HANDLED, logMsg); 683 } 684 break; 685 686 case PRIORITY_PARAM: 687 { 688 mPriority = *((int32*) event->pointer); 689 // DO NOT use ::set_thead_priority() to directly alter the node's control 690 // thread priority. BMediaEventLooper tracks the priority itself and recalculates 691 // the node's scheduling latency whenever SetPriority() is called. This is VERY 692 // important for correct functioning of a node chain. You should *only* alter a 693 // BMediaEventLooper's priority by calling its SetPriority() method. 694 SetPriority(mPriority); 695 696 mLastPrioChange = logMsg.now; 697 BroadcastNewParameterValue(logMsg.now, param, &mPriority, sizeof(mPriority)); 698 logMsg.param.value = (float) mPriority; 699 mLogger->Log(LOG_SET_PARAM_HANDLED, logMsg); 700 } 701 break; 702 703 // log the fact that we "handled" a "set parameter" event for a 704 // nonexistent parameter 705 default: 706 mLogger->Log(LOG_INVALID_PARAM_HANDLED, logMsg); 707 break; 708 } 709 } 710 break; 711 712 case BTimedEventQueue::B_START: 713 // okay, let's go! 714 mLogger->Log(LOG_START_HANDLED, logMsg); 715 break; 716 717 case BTimedEventQueue::B_STOP: 718 mLogger->Log(LOG_STOP_HANDLED, logMsg); 719 // stopping implies not handling any more buffers. So, we flush all pending 720 // buffers out of the event queue before returning to the event loop. 721 EventQueue()->FlushEvents(0, BTimedEventQueue::B_ALWAYS, true, BTimedEventQueue::B_HANDLE_BUFFER); 722 break; 723 724 case BTimedEventQueue::B_SEEK: 725 // seeking the log doesn't make any sense, so we just log that we handled the seek 726 // and return without doing anything else 727 mLogger->Log(LOG_SEEK_HANDLED, logMsg); 728 break; 729 730 case BTimedEventQueue::B_WARP: 731 // similarly, time warps aren't meaningful to the logger, so just record it and return 732 mLogger->Log(LOG_WARP_HANDLED, logMsg); 733 break; 734 735 case BTimedEventQueue::B_DATA_STATUS: 736 // we really don't care about the producer's data status, but this is where 737 // we'd do something about it if we did. 738 logMsg.data_status.status = event->data; 739 mLogger->Log(LOG_DATA_STATUS_HANDLED, logMsg); 740 break; 741 742 default: 743 // hmm, someone enqueued a message that we don't understand. log and ignore it. 744 logMsg.unknown.what = event->type; 745 mLogger->Log(LOG_HANDLE_UNKNOWN, logMsg); 746 break; 747 } 748 } 749