152a38012Sejakowatz /*********************************************************************** 252a38012Sejakowatz * AUTHOR: Marcus Overhagen 352a38012Sejakowatz * FILE: MediaRoster.cpp 452a38012Sejakowatz * DESCR: 552a38012Sejakowatz ***********************************************************************/ 652a38012Sejakowatz #include <MediaRoster.h> 752a38012Sejakowatz #include <Locker.h> 852a38012Sejakowatz #include <Message.h> 952a38012Sejakowatz #include <Messenger.h> 1052a38012Sejakowatz #include <StopWatch.h> 1152a38012Sejakowatz #include <OS.h> 1252a38012Sejakowatz #include <String.h> 1352a38012Sejakowatz #include <TimeSource.h> 1452a38012Sejakowatz #include "debug.h" 1576669a29Sbeveloper #include "TList.h" 1652a38012Sejakowatz #include "PortPool.h" 171299bfb2Sbeveloper #include "SystemTimeSource.h" 180a483e72SIngo Weinhold #include "ServerInterface.h" 19f4925104Sbeveloper #include "DataExchange.h" 2016b7eea4Sbeveloper #include "DormantNodeManager.h" 21f4925104Sbeveloper #include "Notifications.h" 2252a38012Sejakowatz 23f4925104Sbeveloper namespace BPrivate { namespace media { 24f4925104Sbeveloper extern team_id team; 253620737cSbeveloper } } // BPrivate::media 263620737cSbeveloper 273620737cSbeveloper using namespace BPrivate::media; 2852a38012Sejakowatz 2952a38012Sejakowatz // the BMediaRoster destructor is private, 3052a38012Sejakowatz // but _DefaultDeleter is a friend class of 3152a38012Sejakowatz // the BMediaRoster an thus can delete it 3252a38012Sejakowatz class _DefaultDeleter 3352a38012Sejakowatz { 3452a38012Sejakowatz public: 356396865dSbeveloper ~_DefaultDeleter() 366396865dSbeveloper { 376396865dSbeveloper if (BMediaRoster::_sDefault) { 386396865dSbeveloper BMediaRoster::_sDefault->Lock(); 396396865dSbeveloper BMediaRoster::_sDefault->Quit(); 406396865dSbeveloper } 416396865dSbeveloper } 426396865dSbeveloper } _deleter; 4352a38012Sejakowatz 443620737cSbeveloper namespace BPrivate { namespace media { namespace mediaroster { 4552a38012Sejakowatz 4652a38012Sejakowatz status_t GetNode(node_type type, media_node * out_node, int32 * out_input_id = NULL, BString * out_input_name = NULL); 4752a38012Sejakowatz status_t SetNode(node_type type, const media_node *node, const dormant_node_info *info = NULL, const media_input *input = NULL); 4876669a29Sbeveloper status_t GetAllOutputs(const media_node & node, List<media_output> *list); 4976669a29Sbeveloper status_t GetAllInputs(const media_node & node, List<media_input> *list); 5076669a29Sbeveloper status_t PublishOutputs(const media_node & node, List<media_output> *list); 5176669a29Sbeveloper status_t PublishInputs(const media_node & node, List<media_input> *list); 5252a38012Sejakowatz 533620737cSbeveloper status_t 543620737cSbeveloper GetNode(node_type type, media_node * out_node, int32 * out_input_id, BString * out_input_name) 5552a38012Sejakowatz { 5652a38012Sejakowatz if (out_node == NULL) 5752a38012Sejakowatz return B_BAD_VALUE; 5852a38012Sejakowatz 598c6a6096Sbeveloper server_get_node_request request; 608c6a6096Sbeveloper server_get_node_reply reply; 6152a38012Sejakowatz status_t rv; 6252a38012Sejakowatz 638c6a6096Sbeveloper request.type = type; 649e9417d2Sbeveloper request.team = team; 658c6a6096Sbeveloper rv = QueryServer(SERVER_GET_NODE, &request, sizeof(request), &reply, sizeof(reply)); 668c6a6096Sbeveloper if (rv != B_OK) 678c6a6096Sbeveloper return rv; 6852a38012Sejakowatz 6952a38012Sejakowatz *out_node = reply.node; 7052a38012Sejakowatz if (out_input_id) 7152a38012Sejakowatz *out_input_id = reply.input_id; 7252a38012Sejakowatz if (out_input_name) 7352a38012Sejakowatz *out_input_name = reply.input_name; 748c6a6096Sbeveloper return rv; 7552a38012Sejakowatz } 7652a38012Sejakowatz 773620737cSbeveloper status_t 783620737cSbeveloper SetNode(node_type type, const media_node *node, const dormant_node_info *info, const media_input *input) 7952a38012Sejakowatz { 808c6a6096Sbeveloper server_set_node_request request; 818c6a6096Sbeveloper server_set_node_reply reply; 8252a38012Sejakowatz 838c6a6096Sbeveloper request.type = type; 848c6a6096Sbeveloper request.use_node = node ? true : false; 8552a38012Sejakowatz if (node) 868c6a6096Sbeveloper request.node = *node; 878c6a6096Sbeveloper request.use_dni = info ? true : false; 8852a38012Sejakowatz if (info) 898c6a6096Sbeveloper request.dni = *info; 908c6a6096Sbeveloper request.use_input = input ? true : false; 9152a38012Sejakowatz if (input) 928c6a6096Sbeveloper request.input = *input; 9352a38012Sejakowatz 948c6a6096Sbeveloper return QueryServer(SERVER_SET_NODE, &request, sizeof(request), &reply, sizeof(reply)); 9552a38012Sejakowatz } 9652a38012Sejakowatz 973620737cSbeveloper status_t 9876669a29Sbeveloper GetAllOutputs(const media_node & node, List<media_output> *list) 993620737cSbeveloper { 1003620737cSbeveloper int32 cookie; 1013620737cSbeveloper status_t rv; 1023620737cSbeveloper status_t result; 1033620737cSbeveloper 1043620737cSbeveloper result = B_OK; 1053620737cSbeveloper cookie = 0; 10676669a29Sbeveloper list->MakeEmpty(); 1073620737cSbeveloper for (;;) { 1083620737cSbeveloper producer_get_next_output_request request; 1093620737cSbeveloper producer_get_next_output_reply reply; 1103620737cSbeveloper request.cookie = cookie; 1110caff283Sbeveloper rv = QueryPort(node.port, PRODUCER_GET_NEXT_OUTPUT, &request, sizeof(request), &reply, sizeof(reply)); 1123620737cSbeveloper if (rv != B_OK) 1133620737cSbeveloper break; 1143620737cSbeveloper cookie = reply.cookie; 11576669a29Sbeveloper if (!list->Insert(reply.output)) { 11676669a29Sbeveloper FATAL("GetAllOutputs: list->Insert failed\n"); 1173620737cSbeveloper result = B_ERROR; 1183620737cSbeveloper } 1193620737cSbeveloper } 1203620737cSbeveloper 1213620737cSbeveloper producer_dispose_output_cookie_request request; 1223620737cSbeveloper producer_dispose_output_cookie_reply reply; 1230caff283Sbeveloper QueryPort(node.port, PRODUCER_DISPOSE_OUTPUT_COOKIE, &request, sizeof(request), &reply, sizeof(reply)); 1243620737cSbeveloper 1253620737cSbeveloper return result; 1263620737cSbeveloper } 1273620737cSbeveloper 1283620737cSbeveloper status_t 12976669a29Sbeveloper GetAllInputs(const media_node & node, List<media_input> *list) 1303620737cSbeveloper { 1313620737cSbeveloper int32 cookie; 1323620737cSbeveloper status_t rv; 1333620737cSbeveloper status_t result; 1343620737cSbeveloper 1353620737cSbeveloper result = B_OK; 1363620737cSbeveloper cookie = 0; 13776669a29Sbeveloper list->MakeEmpty(); 1383620737cSbeveloper for (;;) { 1393620737cSbeveloper consumer_get_next_input_request request; 1403620737cSbeveloper consumer_get_next_input_reply reply; 1413620737cSbeveloper request.cookie = cookie; 1420caff283Sbeveloper rv = QueryPort(node.port, CONSUMER_GET_NEXT_INPUT, &request, sizeof(request), &reply, sizeof(reply)); 1433620737cSbeveloper if (rv != B_OK) 1443620737cSbeveloper break; 1453620737cSbeveloper cookie = reply.cookie; 14676669a29Sbeveloper if (!list->Insert(reply.input)) { 14776669a29Sbeveloper FATAL("GetAllInputs: list->Insert failed\n"); 1483620737cSbeveloper result = B_ERROR; 1493620737cSbeveloper } 1503620737cSbeveloper } 1513620737cSbeveloper 1523620737cSbeveloper consumer_dispose_input_cookie_request request; 1533620737cSbeveloper consumer_dispose_input_cookie_reply reply; 1540caff283Sbeveloper QueryPort(node.port, CONSUMER_DISPOSE_INPUT_COOKIE, &request, sizeof(request), &reply, sizeof(reply)); 1553620737cSbeveloper 1563620737cSbeveloper return result; 1573620737cSbeveloper } 1583620737cSbeveloper 1593620737cSbeveloper status_t 16076669a29Sbeveloper PublishOutputs(const media_node & node, List<media_output> *list) 1613620737cSbeveloper { 1623620737cSbeveloper server_publish_outputs_request request; 1633620737cSbeveloper server_publish_outputs_reply reply; 1643620737cSbeveloper media_output *output; 1653620737cSbeveloper media_output *outputs; 1663620737cSbeveloper int32 count; 167a7b41a96Sbeveloper status_t rv; 1683620737cSbeveloper 16976669a29Sbeveloper count = list->CountItems(); 1703620737cSbeveloper TRACE("PublishOutputs: publishing %ld\n", count); 1713620737cSbeveloper 1723620737cSbeveloper request.node = node; 1733620737cSbeveloper request.count = count; 1743620737cSbeveloper if (count > MAX_OUTPUTS) { 1753620737cSbeveloper void *start_addr; 1763620737cSbeveloper size_t size; 1773620737cSbeveloper size = ((count * sizeof(media_output)) + B_PAGE_SIZE - 1) & ~(B_PAGE_SIZE - 1); 1783620737cSbeveloper request.area = create_area("publish outputs", &start_addr, B_ANY_ADDRESS, size, B_NO_LOCK, B_READ_AREA | B_WRITE_AREA); 1793620737cSbeveloper if (request.area < B_OK) { 180353b9f6bSbeveloper FATAL("PublishOutputs: failed to create area, %#lx\n", request.area); 1813620737cSbeveloper return (status_t)request.area; 1823620737cSbeveloper } 1833620737cSbeveloper outputs = static_cast<media_output *>(start_addr); 1843620737cSbeveloper } else { 1853620737cSbeveloper request.area = -1; 1863620737cSbeveloper outputs = request.outputs; 1873620737cSbeveloper } 18876669a29Sbeveloper TRACE("PublishOutputs: area %ld\n", request.area); 1893620737cSbeveloper 19076669a29Sbeveloper int i; 19176669a29Sbeveloper for (i = 0, list->Rewind(); list->GetNext(&output); i++) { 19276669a29Sbeveloper ASSERT(i < count); 1933620737cSbeveloper outputs[i] = *output; 1943620737cSbeveloper } 1953620737cSbeveloper 196a7b41a96Sbeveloper rv = QueryServer(SERVER_PUBLISH_OUTPUTS, &request, sizeof(request), &reply, sizeof(reply)); 197a7b41a96Sbeveloper 198a7b41a96Sbeveloper if (request.area != -1) 199a7b41a96Sbeveloper delete_area(request.area); 200a7b41a96Sbeveloper 201a7b41a96Sbeveloper return rv; 2023620737cSbeveloper } 2033620737cSbeveloper 2043620737cSbeveloper status_t 20576669a29Sbeveloper PublishInputs(const media_node & node, List<media_input> *list) 2063620737cSbeveloper { 2073620737cSbeveloper server_publish_inputs_request request; 2083620737cSbeveloper server_publish_inputs_reply reply; 2093620737cSbeveloper media_input *input; 2103620737cSbeveloper media_input *inputs; 2113620737cSbeveloper int32 count; 212a7b41a96Sbeveloper status_t rv; 2133620737cSbeveloper 21476669a29Sbeveloper count = list->CountItems(); 2153620737cSbeveloper TRACE("PublishInputs: publishing %ld\n", count); 2163620737cSbeveloper 2173620737cSbeveloper request.node = node; 2183620737cSbeveloper request.count = count; 2193620737cSbeveloper if (count > MAX_INPUTS) { 2203620737cSbeveloper void *start_addr; 2213620737cSbeveloper size_t size; 2223620737cSbeveloper size = ((count * sizeof(media_input)) + B_PAGE_SIZE - 1) & ~(B_PAGE_SIZE - 1); 2233620737cSbeveloper request.area = create_area("publish inputs", &start_addr, B_ANY_ADDRESS, size, B_NO_LOCK, B_READ_AREA | B_WRITE_AREA); 2243620737cSbeveloper if (request.area < B_OK) { 225353b9f6bSbeveloper FATAL("PublishInputs: failed to create area, %#lx\n", request.area); 2263620737cSbeveloper return (status_t)request.area; 2273620737cSbeveloper } 2283620737cSbeveloper inputs = static_cast<media_input *>(start_addr); 2293620737cSbeveloper } else { 2303620737cSbeveloper request.area = -1; 2313620737cSbeveloper inputs = request.inputs; 2323620737cSbeveloper } 23376669a29Sbeveloper TRACE("PublishInputs: area %ld\n", request.area); 2343620737cSbeveloper 23576669a29Sbeveloper int i; 23676669a29Sbeveloper for (i = 0, list->Rewind(); list->GetNext(&input); i++) { 23776669a29Sbeveloper ASSERT(i < count); 2383620737cSbeveloper inputs[i] = *input; 2393620737cSbeveloper } 2403620737cSbeveloper 241a7b41a96Sbeveloper rv = QueryServer(SERVER_PUBLISH_INPUTS, &request, sizeof(request), &reply, sizeof(reply)); 242a7b41a96Sbeveloper 243a7b41a96Sbeveloper if (request.area != -1) 244a7b41a96Sbeveloper delete_area(request.area); 245a7b41a96Sbeveloper 246a7b41a96Sbeveloper return rv; 2473620737cSbeveloper } 2483620737cSbeveloper 2493620737cSbeveloper } } } // namespace BPrivate::media::mediaroster 2503620737cSbeveloper 2513620737cSbeveloper using namespace BPrivate::media::mediaroster; 25252a38012Sejakowatz 25352a38012Sejakowatz /************************************************************* 25452a38012Sejakowatz * public BMediaRoster 25552a38012Sejakowatz *************************************************************/ 25652a38012Sejakowatz 25752a38012Sejakowatz status_t 25852a38012Sejakowatz BMediaRoster::GetVideoInput(media_node * out_node) 25952a38012Sejakowatz { 26052a38012Sejakowatz CALLED(); 2613620737cSbeveloper return GetNode(VIDEO_INPUT, out_node); 26252a38012Sejakowatz } 26352a38012Sejakowatz 26452a38012Sejakowatz 26552a38012Sejakowatz status_t 26652a38012Sejakowatz BMediaRoster::GetAudioInput(media_node * out_node) 26752a38012Sejakowatz { 26852a38012Sejakowatz CALLED(); 2693620737cSbeveloper return GetNode(AUDIO_INPUT, out_node); 27052a38012Sejakowatz } 27152a38012Sejakowatz 27252a38012Sejakowatz 27352a38012Sejakowatz status_t 27452a38012Sejakowatz BMediaRoster::GetVideoOutput(media_node * out_node) 27552a38012Sejakowatz { 27652a38012Sejakowatz CALLED(); 2773620737cSbeveloper return GetNode(VIDEO_OUTPUT, out_node); 27852a38012Sejakowatz } 27952a38012Sejakowatz 28052a38012Sejakowatz 28152a38012Sejakowatz status_t 28252a38012Sejakowatz BMediaRoster::GetAudioMixer(media_node * out_node) 28352a38012Sejakowatz { 28452a38012Sejakowatz CALLED(); 2853620737cSbeveloper return GetNode(AUDIO_MIXER, out_node); 28652a38012Sejakowatz } 28752a38012Sejakowatz 28852a38012Sejakowatz 28952a38012Sejakowatz status_t 29052a38012Sejakowatz BMediaRoster::GetAudioOutput(media_node * out_node) 29152a38012Sejakowatz { 29252a38012Sejakowatz CALLED(); 2933620737cSbeveloper return GetNode(AUDIO_OUTPUT, out_node); 29452a38012Sejakowatz } 29552a38012Sejakowatz 29652a38012Sejakowatz 29752a38012Sejakowatz status_t 29852a38012Sejakowatz BMediaRoster::GetAudioOutput(media_node * out_node, 29952a38012Sejakowatz int32 * out_input_id, 30052a38012Sejakowatz BString * out_input_name) 30152a38012Sejakowatz { 30252a38012Sejakowatz CALLED(); 3033620737cSbeveloper return GetNode(AUDIO_OUTPUT_EX, out_node, out_input_id, out_input_name); 30452a38012Sejakowatz } 30552a38012Sejakowatz 30652a38012Sejakowatz 30752a38012Sejakowatz status_t 30852a38012Sejakowatz BMediaRoster::GetTimeSource(media_node * out_node) 30952a38012Sejakowatz { 31052a38012Sejakowatz CALLED(); 3113620737cSbeveloper return GetNode(TIME_SOURCE, out_node); 31252a38012Sejakowatz } 31352a38012Sejakowatz 31452a38012Sejakowatz 31552a38012Sejakowatz status_t 31652a38012Sejakowatz BMediaRoster::SetVideoInput(const media_node & producer) 31752a38012Sejakowatz { 31852a38012Sejakowatz CALLED(); 3193620737cSbeveloper return SetNode(VIDEO_INPUT, &producer); 32052a38012Sejakowatz } 32152a38012Sejakowatz 32252a38012Sejakowatz 32352a38012Sejakowatz status_t 32452a38012Sejakowatz BMediaRoster::SetVideoInput(const dormant_node_info & producer) 32552a38012Sejakowatz { 32652a38012Sejakowatz CALLED(); 3273620737cSbeveloper return SetNode(VIDEO_INPUT, NULL, &producer); 32852a38012Sejakowatz } 32952a38012Sejakowatz 33052a38012Sejakowatz 33152a38012Sejakowatz status_t 33252a38012Sejakowatz BMediaRoster::SetAudioInput(const media_node & producer) 33352a38012Sejakowatz { 33452a38012Sejakowatz CALLED(); 3353620737cSbeveloper return SetNode(AUDIO_INPUT, &producer); 33652a38012Sejakowatz } 33752a38012Sejakowatz 33852a38012Sejakowatz 33952a38012Sejakowatz status_t 34052a38012Sejakowatz BMediaRoster::SetAudioInput(const dormant_node_info & producer) 34152a38012Sejakowatz { 34252a38012Sejakowatz CALLED(); 3433620737cSbeveloper return SetNode(AUDIO_INPUT, NULL, &producer); 34452a38012Sejakowatz } 34552a38012Sejakowatz 34652a38012Sejakowatz 34752a38012Sejakowatz status_t 34852a38012Sejakowatz BMediaRoster::SetVideoOutput(const media_node & consumer) 34952a38012Sejakowatz { 35052a38012Sejakowatz CALLED(); 3513620737cSbeveloper return SetNode(VIDEO_OUTPUT, &consumer); 35252a38012Sejakowatz } 35352a38012Sejakowatz 35452a38012Sejakowatz 35552a38012Sejakowatz status_t 35652a38012Sejakowatz BMediaRoster::SetVideoOutput(const dormant_node_info & consumer) 35752a38012Sejakowatz { 35852a38012Sejakowatz CALLED(); 3593620737cSbeveloper return SetNode(VIDEO_OUTPUT, NULL, &consumer); 36052a38012Sejakowatz } 36152a38012Sejakowatz 36252a38012Sejakowatz 36352a38012Sejakowatz status_t 36452a38012Sejakowatz BMediaRoster::SetAudioOutput(const media_node & consumer) 36552a38012Sejakowatz { 36652a38012Sejakowatz CALLED(); 3673620737cSbeveloper return SetNode(AUDIO_OUTPUT, &consumer); 36852a38012Sejakowatz } 36952a38012Sejakowatz 37052a38012Sejakowatz 37152a38012Sejakowatz status_t 37252a38012Sejakowatz BMediaRoster::SetAudioOutput(const media_input & input_to_output) 37352a38012Sejakowatz { 37452a38012Sejakowatz CALLED(); 3753620737cSbeveloper return SetNode(AUDIO_OUTPUT, NULL, NULL, &input_to_output); 37652a38012Sejakowatz } 37752a38012Sejakowatz 37852a38012Sejakowatz 37952a38012Sejakowatz status_t 38052a38012Sejakowatz BMediaRoster::SetAudioOutput(const dormant_node_info & consumer) 38152a38012Sejakowatz { 38252a38012Sejakowatz CALLED(); 3833620737cSbeveloper return SetNode(AUDIO_OUTPUT, NULL, &consumer); 38452a38012Sejakowatz } 38552a38012Sejakowatz 38652a38012Sejakowatz 38752a38012Sejakowatz status_t 38852a38012Sejakowatz BMediaRoster::GetNodeFor(media_node_id node, 38952a38012Sejakowatz media_node * clone) 39052a38012Sejakowatz { 3919e9417d2Sbeveloper CALLED(); 3929e9417d2Sbeveloper if (clone == NULL) 3939e9417d2Sbeveloper return B_BAD_VALUE; 3949e9417d2Sbeveloper if (node <= 0) 3959e9417d2Sbeveloper return B_MEDIA_BAD_NODE; 3969e9417d2Sbeveloper 3979e9417d2Sbeveloper server_get_node_for_request request; 3989e9417d2Sbeveloper server_get_node_for_reply reply; 3999e9417d2Sbeveloper status_t rv; 4009e9417d2Sbeveloper 4019e9417d2Sbeveloper request.nodeid = node; 4029e9417d2Sbeveloper request.team = team; 4039e9417d2Sbeveloper 4049e9417d2Sbeveloper rv = QueryServer(SERVER_GET_NODE_FOR, &request, sizeof(request), &reply, sizeof(reply)); 4059e9417d2Sbeveloper if (rv != B_OK) 4069e9417d2Sbeveloper return rv; 4079e9417d2Sbeveloper 4089e9417d2Sbeveloper *clone = reply.clone; 4099e9417d2Sbeveloper return B_OK; 41052a38012Sejakowatz } 41152a38012Sejakowatz 41252a38012Sejakowatz 41352a38012Sejakowatz status_t 41452a38012Sejakowatz BMediaRoster::GetSystemTimeSource(media_node * clone) 41552a38012Sejakowatz { 41652a38012Sejakowatz CALLED(); 4173620737cSbeveloper return GetNode(SYSTEM_TIME_SOURCE, clone); 41852a38012Sejakowatz } 41952a38012Sejakowatz 42052a38012Sejakowatz 42152a38012Sejakowatz status_t 42252a38012Sejakowatz BMediaRoster::ReleaseNode(const media_node & node) 42352a38012Sejakowatz { 4249e9417d2Sbeveloper CALLED(); 4259e9417d2Sbeveloper if (node.node <= 0) 4269e9417d2Sbeveloper return B_MEDIA_BAD_NODE; 42752a38012Sejakowatz 4289e9417d2Sbeveloper server_release_node_request request; 4299e9417d2Sbeveloper server_release_node_reply reply; 4309e9417d2Sbeveloper 4319e9417d2Sbeveloper request.node = node; 4329e9417d2Sbeveloper request.team = team; 4339e9417d2Sbeveloper 4349e9417d2Sbeveloper return QueryServer(SERVER_RELEASE_NODE, &request, sizeof(request), &reply, sizeof(reply)); 4359e9417d2Sbeveloper } 43652a38012Sejakowatz 43752a38012Sejakowatz 43852a38012Sejakowatz BTimeSource * 43952a38012Sejakowatz BMediaRoster::MakeTimeSourceFor(const media_node & for_node) 44052a38012Sejakowatz { 4411299bfb2Sbeveloper BROKEN(); 4421299bfb2Sbeveloper return new _SysTimeSource(); // XXX fix this 44352a38012Sejakowatz } 44452a38012Sejakowatz 44552a38012Sejakowatz 44652a38012Sejakowatz status_t 44752a38012Sejakowatz BMediaRoster::Connect(const media_source & from, 44852a38012Sejakowatz const media_destination & to, 44952a38012Sejakowatz media_format * io_format, 45052a38012Sejakowatz media_output * out_output, 45152a38012Sejakowatz media_input * out_input) 45252a38012Sejakowatz { 453eae26d3dSbeveloper return BMediaRoster::Connect(from, to, io_format, out_output, out_input, 0); 454eae26d3dSbeveloper } 455eae26d3dSbeveloper 456eae26d3dSbeveloper 457eae26d3dSbeveloper status_t 458eae26d3dSbeveloper BMediaRoster::Connect(const media_source & from, 459eae26d3dSbeveloper const media_destination & to, 460eae26d3dSbeveloper media_format * io_format, 461eae26d3dSbeveloper media_output * out_output, 462eae26d3dSbeveloper media_input * out_input, 463eae26d3dSbeveloper uint32 in_flags, 464eae26d3dSbeveloper void * _reserved) 465eae26d3dSbeveloper { 46652a38012Sejakowatz CALLED(); 46752a38012Sejakowatz if (io_format == NULL || out_output == NULL || out_input == NULL) 46852a38012Sejakowatz return B_BAD_VALUE; 4699e9417d2Sbeveloper if (from == media_source::null) { 470353b9f6bSbeveloper FATAL("BMediaRoster::Connect: media_source invalid\n"); 47152a38012Sejakowatz return B_MEDIA_BAD_SOURCE; 4729e9417d2Sbeveloper } 4739e9417d2Sbeveloper if (to == media_destination::null) { 474353b9f6bSbeveloper FATAL("BMediaRoster::Connect: media_destination invalid\n"); 47552a38012Sejakowatz return B_MEDIA_BAD_DESTINATION; 4769e9417d2Sbeveloper } 47752a38012Sejakowatz 47852a38012Sejakowatz status_t rv; 4798c6a6096Sbeveloper producer_format_proposal_request request1; 4808c6a6096Sbeveloper producer_format_proposal_reply reply1; 48152a38012Sejakowatz 48252a38012Sejakowatz // BBufferProducer::FormatProposal 4838c6a6096Sbeveloper request1.output = from; 4848c6a6096Sbeveloper request1.format = *io_format; 4858c6a6096Sbeveloper rv = QueryPort(from.port, PRODUCER_FORMAT_PROPOSAL, &request1, sizeof(request1), &reply1, sizeof(reply1)); 4868c6a6096Sbeveloper if (rv != B_OK) { 487353b9f6bSbeveloper FATAL("BMediaRoster::Connect: aborted after BBufferProducer::FormatProposal, status = %#lx\n",rv); 48852a38012Sejakowatz return rv; 48952a38012Sejakowatz } 4908c6a6096Sbeveloper // reply1.format now contains the format proposed by the producer 4918c6a6096Sbeveloper 4928c6a6096Sbeveloper consumer_accept_format_request request2; 4938c6a6096Sbeveloper consumer_accept_format_reply reply2; 4948c6a6096Sbeveloper 4958c6a6096Sbeveloper // BBufferConsumer::AcceptFormat 4968c6a6096Sbeveloper request2.dest = to; 4978c6a6096Sbeveloper request2.format = reply1.format; 4988c6a6096Sbeveloper rv = QueryPort(to.port, CONSUMER_ACCEPT_FORMAT, &request2, sizeof(request2), &reply2, sizeof(reply2)); 4998c6a6096Sbeveloper if (rv != B_OK) { 500353b9f6bSbeveloper FATAL("BMediaRoster::Connect: aborted after BBufferConsumer::AcceptFormat, status = %#lx\n",rv); 5018c6a6096Sbeveloper return rv; 5028c6a6096Sbeveloper } 5038c6a6096Sbeveloper // reply2.format now contains the format accepted by the consumer 5048c6a6096Sbeveloper 5058c6a6096Sbeveloper // BBufferProducer::PrepareToConnect 5068c6a6096Sbeveloper producer_prepare_to_connect_request request3; 5078c6a6096Sbeveloper producer_prepare_to_connect_reply reply3; 5088c6a6096Sbeveloper 5098c6a6096Sbeveloper request3.source = from; 5108c6a6096Sbeveloper request3.destination = to; 5118c6a6096Sbeveloper request3.format = reply2.format; 5128c6a6096Sbeveloper strcpy(request3.name, "XXX some default name"); // XXX fix this 5138c6a6096Sbeveloper rv = QueryPort(from.port, PRODUCER_PREPARE_TO_CONNECT, &request3, sizeof(request3), &reply3, sizeof(reply3)); 5148c6a6096Sbeveloper if (rv != B_OK) { 515353b9f6bSbeveloper FATAL("BMediaRoster::Connect: aborted after BBufferProducer::PrepareToConnect, status = %#lx\n",rv); 5168c6a6096Sbeveloper return rv; 5178c6a6096Sbeveloper } 5188c6a6096Sbeveloper // reply3.format is still our pretty media format 5198c6a6096Sbeveloper // reply3.out_source the real source to be used for the connection 5208c6a6096Sbeveloper // reply3.name the name BBufferConsumer::Connected will see in the outInput->name argument 5218c6a6096Sbeveloper 5228c6a6096Sbeveloper // BBufferConsumer::Connected 5238c6a6096Sbeveloper consumer_connected_request request4; 5248c6a6096Sbeveloper consumer_connected_reply reply4; 5258c6a6096Sbeveloper status_t con_status; 5268c6a6096Sbeveloper 5278c6a6096Sbeveloper request4.producer = reply3.out_source; 5288c6a6096Sbeveloper request4.where = to; 5298c6a6096Sbeveloper request4.with_format = reply3.format; 5308c6a6096Sbeveloper con_status = QueryPort(to.port, CONSUMER_CONNECTED, &request4, sizeof(request4), &reply4, sizeof(reply4)); 5318c6a6096Sbeveloper if (con_status != B_OK) { 532353b9f6bSbeveloper FATAL("BMediaRoster::Connect: aborting after BBufferConsumer::Connected, status = %#lx\n",con_status); 5338c6a6096Sbeveloper // we do NOT return here! 5348c6a6096Sbeveloper } 5358c6a6096Sbeveloper // con_status contains the status code to be supplied to BBufferProducer::Connect's status argument 5368c6a6096Sbeveloper // reply4.input contains the media_input that describes the connection from the consumer point of view 5378c6a6096Sbeveloper 5388c6a6096Sbeveloper // BBufferProducer::Connect 5398c6a6096Sbeveloper producer_connect_request request5; 5408c6a6096Sbeveloper producer_connect_reply reply5; 5418c6a6096Sbeveloper 5428c6a6096Sbeveloper request5.error = con_status; 5438c6a6096Sbeveloper request5.source = reply3.out_source; 5448c6a6096Sbeveloper request5.destination = reply4.input.destination; 5458c6a6096Sbeveloper request5.format = reply3.format; // XXX reply4.input.format ??? 5468c6a6096Sbeveloper strcpy(request5.name, reply4.input.name); 5470caff283Sbeveloper rv = QueryPort(reply4.input.source.port, PRODUCER_CONNECT, &request5, sizeof(request5), &reply5, sizeof(reply5)); 5488c6a6096Sbeveloper if (con_status != B_OK) { 549353b9f6bSbeveloper FATAL("BMediaRoster::Connect: aborted\n"); 5508c6a6096Sbeveloper return con_status; 5518c6a6096Sbeveloper } 5528c6a6096Sbeveloper if (rv != B_OK) { 553353b9f6bSbeveloper FATAL("BMediaRoster::Connect: aborted after BBufferProducer::Connect, status = %#lx\n",rv); 5548c6a6096Sbeveloper return rv; 5558c6a6096Sbeveloper } 5568c6a6096Sbeveloper // reply5.name contains the name assigned to the connection by the producer 5578c6a6096Sbeveloper 5588c6a6096Sbeveloper // find the output node 5598c6a6096Sbeveloper // XXX isn't there a easier way? 5608c6a6096Sbeveloper media_node sourcenode; 5618c6a6096Sbeveloper GetNodeFor(NodeIDFor(from.port), &sourcenode); 5628c6a6096Sbeveloper ReleaseNode(sourcenode); 5638c6a6096Sbeveloper 5648c6a6096Sbeveloper // initilize connection info 5658c6a6096Sbeveloper *io_format = reply3.format; 5668c6a6096Sbeveloper *out_input = reply4.input; 5678c6a6096Sbeveloper out_output->node = sourcenode; 5688c6a6096Sbeveloper out_output->source = reply4.input.source; 5698c6a6096Sbeveloper out_output->destination = reply4.input.destination; 5708c6a6096Sbeveloper out_output->format = reply4.input.format; 5718c6a6096Sbeveloper strcpy(out_output->name, reply5.name); 5728c6a6096Sbeveloper 5738c6a6096Sbeveloper // the connection is now made 5748c6a6096Sbeveloper 5758c6a6096Sbeveloper 5768c6a6096Sbeveloper // XXX register connection with server 5779e9417d2Sbeveloper // XXX we should just send a notification, instead of republishing all endpoints 57876669a29Sbeveloper List<media_output> outlist; 57976669a29Sbeveloper List<media_input> inlist; 58076669a29Sbeveloper if (B_OK == GetAllOutputs(out_output->node , &outlist)) 58176669a29Sbeveloper PublishOutputs(out_output->node , &outlist); 58276669a29Sbeveloper if (B_OK == GetAllInputs(out_input->node , &inlist)) 58376669a29Sbeveloper PublishInputs(out_input->node, &inlist); 5848c6a6096Sbeveloper 5858c6a6096Sbeveloper 5868c6a6096Sbeveloper // XXX if (mute) BBufferProducer::EnableOutput(false) 5879e9417d2Sbeveloper if (in_flags & B_CONNECT_MUTED) { 5889e9417d2Sbeveloper } 5898c6a6096Sbeveloper 5908c6a6096Sbeveloper 5919e9417d2Sbeveloper // send a notification 5929e9417d2Sbeveloper BPrivate::media::notifications::ConnectionMade(*out_input, *out_output, *io_format); 5939e9417d2Sbeveloper 5948c6a6096Sbeveloper return B_OK; 5958c6a6096Sbeveloper }; 59652a38012Sejakowatz 59752a38012Sejakowatz 59852a38012Sejakowatz status_t 5999e9417d2Sbeveloper BMediaRoster::Disconnect(media_node_id source_nodeid, 60052a38012Sejakowatz const media_source & source, 6019e9417d2Sbeveloper media_node_id destination_nodeid, 60252a38012Sejakowatz const media_destination & destination) 60352a38012Sejakowatz { 6049e9417d2Sbeveloper CALLED(); 6059e9417d2Sbeveloper if (source_nodeid <= 0) { 606353b9f6bSbeveloper FATAL("BMediaRoster::Disconnect: source media_node_id invalid\n"); 6079e9417d2Sbeveloper return B_MEDIA_BAD_SOURCE; 6089e9417d2Sbeveloper } 6099e9417d2Sbeveloper if (destination_nodeid <= 0) { 610353b9f6bSbeveloper FATAL("BMediaRoster::Disconnect: source media_node_id invalid\n"); 6119e9417d2Sbeveloper return B_MEDIA_BAD_DESTINATION; 6129e9417d2Sbeveloper } 6139e9417d2Sbeveloper if (source == media_source::null) { 614353b9f6bSbeveloper FATAL("BMediaRoster::Disconnect: media_source invalid\n"); 6159e9417d2Sbeveloper return B_MEDIA_BAD_SOURCE; 6169e9417d2Sbeveloper } 6179e9417d2Sbeveloper if (destination == media_destination::null) { 618353b9f6bSbeveloper FATAL("BMediaRoster::Disconnect: media_destination invalid\n"); 6199e9417d2Sbeveloper return B_MEDIA_BAD_DESTINATION; 6209e9417d2Sbeveloper } 6219e9417d2Sbeveloper 6229e9417d2Sbeveloper producer_disconnect_request request2; 6239e9417d2Sbeveloper producer_disconnect_reply reply2; 6249e9417d2Sbeveloper consumer_disconnected_request request1; 6259e9417d2Sbeveloper consumer_disconnected_reply reply1; 6269e9417d2Sbeveloper status_t rv1, rv2; 6279e9417d2Sbeveloper 6289e9417d2Sbeveloper // XXX we should ask the server if this connection really exists 6299e9417d2Sbeveloper 6309e9417d2Sbeveloper request1.source = source; 6319e9417d2Sbeveloper request1.destination = destination; 6329e9417d2Sbeveloper request2.source = source; 6339e9417d2Sbeveloper request2.destination = destination; 6349e9417d2Sbeveloper 6359e9417d2Sbeveloper rv1 = QueryPort(source.port, PRODUCER_DISCONNECT, &request1, sizeof(request1), &reply1, sizeof(reply1)); 6369e9417d2Sbeveloper rv2 = QueryPort(destination.port, CONSUMER_DISCONNECTED, &request2, sizeof(request2), &reply2, sizeof(reply2)); 6379e9417d2Sbeveloper 6389e9417d2Sbeveloper // XXX unregister connection with server 6399e9417d2Sbeveloper // XXX we should just send a notification, instead of republishing all endpoints 64076669a29Sbeveloper List<media_output> outlist; 64176669a29Sbeveloper List<media_input> inlist; 6429e9417d2Sbeveloper media_node sourcenode; 6439e9417d2Sbeveloper media_node destnode; 6449e9417d2Sbeveloper if (B_OK == GetNodeFor(source_nodeid, &sourcenode)) { 64576669a29Sbeveloper if (B_OK == GetAllOutputs(sourcenode , &outlist)) 64676669a29Sbeveloper PublishOutputs(sourcenode , &outlist); 6479e9417d2Sbeveloper ReleaseNode(sourcenode); 648353b9f6bSbeveloper } else FATAL("BMediaRoster::Disconnect: source GetNodeFor failed\n"); 6499e9417d2Sbeveloper if (B_OK == GetNodeFor(destination_nodeid, &destnode)) { 65076669a29Sbeveloper if (B_OK == GetAllInputs(destnode , &inlist)) 65176669a29Sbeveloper PublishInputs(destnode, &inlist); 6529e9417d2Sbeveloper ReleaseNode(destnode); 653353b9f6bSbeveloper } else FATAL("BMediaRoster::Disconnect: dest GetNodeFor failed\n"); 6549e9417d2Sbeveloper 6559e9417d2Sbeveloper 6569e9417d2Sbeveloper // send a notification 6579e9417d2Sbeveloper BPrivate::media::notifications::ConnectionBroken(source, destination); 6589e9417d2Sbeveloper 6599e9417d2Sbeveloper return (rv1 != B_OK || rv2 != B_OK) ? B_ERROR : B_OK; 66052a38012Sejakowatz } 66152a38012Sejakowatz 66252a38012Sejakowatz 66352a38012Sejakowatz status_t 66452a38012Sejakowatz BMediaRoster::StartNode(const media_node & node, 66552a38012Sejakowatz bigtime_t at_performance_time) 66652a38012Sejakowatz { 66752a38012Sejakowatz CALLED(); 6689e9417d2Sbeveloper if (node.node <= 0) 66952a38012Sejakowatz return B_MEDIA_BAD_NODE; 67052a38012Sejakowatz 6716396865dSbeveloper node_start_command command; 6726396865dSbeveloper command.performance_time = at_performance_time; 67352a38012Sejakowatz 6746396865dSbeveloper return SendToPort(node.port, NODE_START, &command, sizeof(command)); 67552a38012Sejakowatz } 67652a38012Sejakowatz 67752a38012Sejakowatz 67852a38012Sejakowatz status_t 67952a38012Sejakowatz BMediaRoster::StopNode(const media_node & node, 68052a38012Sejakowatz bigtime_t at_performance_time, 68152a38012Sejakowatz bool immediate) 68252a38012Sejakowatz { 68352a38012Sejakowatz CALLED(); 6849e9417d2Sbeveloper if (node.node <= 0) 68552a38012Sejakowatz return B_MEDIA_BAD_NODE; 68652a38012Sejakowatz 6876396865dSbeveloper node_stop_command command; 6886396865dSbeveloper command.performance_time = at_performance_time; 6896396865dSbeveloper command.immediate = immediate; 69052a38012Sejakowatz 6916396865dSbeveloper return SendToPort(node.port, NODE_STOP, &command, sizeof(command)); 69252a38012Sejakowatz } 69352a38012Sejakowatz 69452a38012Sejakowatz 69552a38012Sejakowatz status_t 69652a38012Sejakowatz BMediaRoster::SeekNode(const media_node & node, 69752a38012Sejakowatz bigtime_t to_media_time, 69852a38012Sejakowatz bigtime_t at_performance_time) 69952a38012Sejakowatz { 70052a38012Sejakowatz CALLED(); 7019e9417d2Sbeveloper if (node.node <= 0) 70252a38012Sejakowatz return B_MEDIA_BAD_NODE; 70352a38012Sejakowatz 7046396865dSbeveloper node_seek_command command; 7056396865dSbeveloper command.media_time = to_media_time; 7066396865dSbeveloper command.performance_time = at_performance_time; 70752a38012Sejakowatz 7086396865dSbeveloper return SendToPort(node.port, NODE_SEEK, &command, sizeof(command)); 70952a38012Sejakowatz } 71052a38012Sejakowatz 71152a38012Sejakowatz 71252a38012Sejakowatz status_t 71352a38012Sejakowatz BMediaRoster::StartTimeSource(const media_node & node, 71452a38012Sejakowatz bigtime_t at_real_time) 71552a38012Sejakowatz { 71652a38012Sejakowatz CALLED(); 7179e9417d2Sbeveloper if (node.node <= 0) 71852a38012Sejakowatz return B_MEDIA_BAD_NODE; 71952a38012Sejakowatz if ((node.kind & B_TIME_SOURCE) == 0) 72052a38012Sejakowatz return B_MEDIA_BAD_NODE; 72152a38012Sejakowatz 72252a38012Sejakowatz BTimeSource::time_source_op_info msg; 72352a38012Sejakowatz msg.op = BTimeSource::B_TIMESOURCE_START; 72452a38012Sejakowatz msg.real_time = at_real_time; 72552a38012Sejakowatz 72652a38012Sejakowatz return write_port(node.port, TIMESOURCE_OP, &msg, sizeof(msg)); 72752a38012Sejakowatz } 72852a38012Sejakowatz 72952a38012Sejakowatz 73052a38012Sejakowatz status_t 73152a38012Sejakowatz BMediaRoster::StopTimeSource(const media_node & node, 73252a38012Sejakowatz bigtime_t at_real_time, 73352a38012Sejakowatz bool immediate) 73452a38012Sejakowatz { 73552a38012Sejakowatz CALLED(); 7369e9417d2Sbeveloper if (node.node <= 0) 73752a38012Sejakowatz return B_MEDIA_BAD_NODE; 73852a38012Sejakowatz if ((node.kind & B_TIME_SOURCE) == 0) 73952a38012Sejakowatz return B_MEDIA_BAD_NODE; 74052a38012Sejakowatz 74152a38012Sejakowatz BTimeSource::time_source_op_info msg; 74252a38012Sejakowatz msg.op = immediate ? BTimeSource::B_TIMESOURCE_STOP_IMMEDIATELY : BTimeSource::B_TIMESOURCE_STOP; 74352a38012Sejakowatz msg.real_time = at_real_time; 74452a38012Sejakowatz 74552a38012Sejakowatz return write_port(node.port, TIMESOURCE_OP, &msg, sizeof(msg)); 74652a38012Sejakowatz } 74752a38012Sejakowatz 74852a38012Sejakowatz 74952a38012Sejakowatz status_t 75052a38012Sejakowatz BMediaRoster::SeekTimeSource(const media_node & node, 75152a38012Sejakowatz bigtime_t to_performance_time, 75252a38012Sejakowatz bigtime_t at_real_time) 75352a38012Sejakowatz { 75452a38012Sejakowatz CALLED(); 7559e9417d2Sbeveloper if (node.node <= 0) 75652a38012Sejakowatz return B_MEDIA_BAD_NODE; 75752a38012Sejakowatz if ((node.kind & B_TIME_SOURCE) == 0) 75852a38012Sejakowatz return B_MEDIA_BAD_NODE; 75952a38012Sejakowatz 76052a38012Sejakowatz BTimeSource::time_source_op_info msg; 76152a38012Sejakowatz msg.op = BTimeSource::B_TIMESOURCE_SEEK; 76252a38012Sejakowatz msg.real_time = at_real_time; 76352a38012Sejakowatz msg.performance_time = to_performance_time; 76452a38012Sejakowatz 76552a38012Sejakowatz return write_port(node.port, TIMESOURCE_OP, &msg, sizeof(msg)); 76652a38012Sejakowatz } 76752a38012Sejakowatz 76852a38012Sejakowatz 76952a38012Sejakowatz status_t 77052a38012Sejakowatz BMediaRoster::SyncToNode(const media_node & node, 77152a38012Sejakowatz bigtime_t at_time, 77252a38012Sejakowatz bigtime_t timeout) 77352a38012Sejakowatz { 77452a38012Sejakowatz UNIMPLEMENTED(); 77552a38012Sejakowatz return B_ERROR; 77652a38012Sejakowatz } 77752a38012Sejakowatz 77852a38012Sejakowatz 77952a38012Sejakowatz status_t 78052a38012Sejakowatz BMediaRoster::SetRunModeNode(const media_node & node, 78152a38012Sejakowatz BMediaNode::run_mode mode) 78252a38012Sejakowatz { 78352a38012Sejakowatz CALLED(); 7849e9417d2Sbeveloper if (node.node <= 0) 78552a38012Sejakowatz return B_MEDIA_BAD_NODE; 78652a38012Sejakowatz 787dfb2ad61Sbeveloper node_set_run_mode_command msg; 78852a38012Sejakowatz msg.mode = mode; 78952a38012Sejakowatz 79052a38012Sejakowatz return write_port(node.port, NODE_SET_RUN_MODE, &msg, sizeof(msg)); 79152a38012Sejakowatz } 79252a38012Sejakowatz 79352a38012Sejakowatz 79452a38012Sejakowatz status_t 79552a38012Sejakowatz BMediaRoster::PrerollNode(const media_node & node) 79652a38012Sejakowatz { 79752a38012Sejakowatz CALLED(); 7989e9417d2Sbeveloper if (node.node <= 0) 79952a38012Sejakowatz return B_MEDIA_BAD_NODE; 80052a38012Sejakowatz 80152a38012Sejakowatz char dummy; 80252a38012Sejakowatz return write_port(node.port, NODE_PREROLL, &dummy, sizeof(dummy)); 80352a38012Sejakowatz } 80452a38012Sejakowatz 80552a38012Sejakowatz 80652a38012Sejakowatz status_t 80752a38012Sejakowatz BMediaRoster::RollNode(const media_node & node, 80852a38012Sejakowatz bigtime_t startPerformance, 80952a38012Sejakowatz bigtime_t stopPerformance, 81052a38012Sejakowatz bigtime_t atMediaTime) 81152a38012Sejakowatz { 81252a38012Sejakowatz UNIMPLEMENTED(); 81352a38012Sejakowatz return B_ERROR; 81452a38012Sejakowatz } 81552a38012Sejakowatz 81652a38012Sejakowatz 81752a38012Sejakowatz status_t 81852a38012Sejakowatz BMediaRoster::SetProducerRunModeDelay(const media_node & node, 81952a38012Sejakowatz bigtime_t delay, 82052a38012Sejakowatz BMediaNode::run_mode mode) 82152a38012Sejakowatz { 82252a38012Sejakowatz UNIMPLEMENTED(); 82352a38012Sejakowatz return B_ERROR; 82452a38012Sejakowatz } 82552a38012Sejakowatz 82652a38012Sejakowatz 82752a38012Sejakowatz status_t 82852a38012Sejakowatz BMediaRoster::SetProducerRate(const media_node & producer, 82952a38012Sejakowatz int32 numer, 83052a38012Sejakowatz int32 denom) 83152a38012Sejakowatz { 83252a38012Sejakowatz CALLED(); 83352a38012Sejakowatz if (producer.node == 0) 83452a38012Sejakowatz return B_MEDIA_BAD_NODE; 83552a38012Sejakowatz if ((producer.kind & B_BUFFER_PRODUCER) == 0) 83652a38012Sejakowatz return B_MEDIA_BAD_NODE; 83752a38012Sejakowatz 838dfb2ad61Sbeveloper producer_set_play_rate_request msg; 839dfb2ad61Sbeveloper producer_set_play_rate_reply reply; 84052a38012Sejakowatz status_t rv; 84152a38012Sejakowatz int32 code; 84252a38012Sejakowatz 84352a38012Sejakowatz msg.numer = numer; 84452a38012Sejakowatz msg.denom = denom; 84552a38012Sejakowatz msg.reply_port = _PortPool->GetPort(); 84652a38012Sejakowatz rv = write_port(producer.node, PRODUCER_SET_PLAY_RATE, &msg, sizeof(msg)); 84752a38012Sejakowatz if (rv != B_OK) { 84852a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 84952a38012Sejakowatz return rv; 85052a38012Sejakowatz } 85152a38012Sejakowatz rv = read_port(msg.reply_port, &code, &reply, sizeof(reply)); 85252a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 85352a38012Sejakowatz return (rv < B_OK) ? rv : reply.result; 85452a38012Sejakowatz } 85552a38012Sejakowatz 85652a38012Sejakowatz 85752a38012Sejakowatz /* Nodes will have available inputs/outputs as long as they are capable */ 85852a38012Sejakowatz /* of accepting more connections. The node may create an additional */ 85952a38012Sejakowatz /* output or input as the currently available is taken into usage. */ 86052a38012Sejakowatz status_t 86152a38012Sejakowatz BMediaRoster::GetLiveNodeInfo(const media_node & node, 86252a38012Sejakowatz live_node_info * out_live_info) 86352a38012Sejakowatz { 8649e9417d2Sbeveloper CALLED(); 8659e9417d2Sbeveloper if (out_live_info == NULL) 8669e9417d2Sbeveloper return B_BAD_VALUE; 8679e9417d2Sbeveloper if (node.node <= 0) 8689e9417d2Sbeveloper return B_MEDIA_BAD_NODE; 8699e9417d2Sbeveloper 8709e9417d2Sbeveloper server_get_live_node_info_request request; 8719e9417d2Sbeveloper server_get_live_node_info_reply reply; 8729e9417d2Sbeveloper status_t rv; 8739e9417d2Sbeveloper 8749e9417d2Sbeveloper request.node = node; 8759e9417d2Sbeveloper 8761299bfb2Sbeveloper rv = QueryServer(SERVER_GET_LIVE_NODE_INFO, &request, sizeof(request), &reply, sizeof(reply)); 8779e9417d2Sbeveloper if (rv != B_OK) 8789e9417d2Sbeveloper return rv; 8799e9417d2Sbeveloper 8809e9417d2Sbeveloper *out_live_info = reply.live_info; 8819e9417d2Sbeveloper return B_OK; 88252a38012Sejakowatz } 88352a38012Sejakowatz 88452a38012Sejakowatz 88552a38012Sejakowatz status_t 88652a38012Sejakowatz BMediaRoster::GetLiveNodes(live_node_info * out_live_nodes, 88752a38012Sejakowatz int32 * io_total_count, 88852a38012Sejakowatz const media_format * has_input, 88952a38012Sejakowatz const media_format * has_output, 89052a38012Sejakowatz const char * name, 89152a38012Sejakowatz uint64 node_kinds) 89252a38012Sejakowatz { 8939e9417d2Sbeveloper CALLED(); 8949e9417d2Sbeveloper if (out_live_nodes == NULL || io_total_count == NULL) 8959e9417d2Sbeveloper return B_BAD_VALUE; 8969e9417d2Sbeveloper if (*io_total_count <= 0) 8979e9417d2Sbeveloper return B_BAD_VALUE; 8989e9417d2Sbeveloper 8999e9417d2Sbeveloper // XXX we also support the wildcard search as GetDormantNodes does. This needs to be documented 9009e9417d2Sbeveloper 9019e9417d2Sbeveloper server_get_live_nodes_request request; 9029e9417d2Sbeveloper server_get_live_nodes_reply reply; 9039e9417d2Sbeveloper status_t rv; 9049e9417d2Sbeveloper 9059e9417d2Sbeveloper request.maxcount = *io_total_count; 9069e9417d2Sbeveloper request.has_input = (bool) has_input; 9079e9417d2Sbeveloper if (has_input) 9089e9417d2Sbeveloper request.inputformat = *has_input; // XXX we should not make a flat copy of media_format 9099e9417d2Sbeveloper request.has_output = (bool) has_output; 9109e9417d2Sbeveloper if (has_output) 9119e9417d2Sbeveloper request.outputformat = *has_output; // XXX we should not make a flat copy of media_format 9129e9417d2Sbeveloper request.has_name = (bool) name; 9139e9417d2Sbeveloper if (name) { 9149e9417d2Sbeveloper int len = strlen(name); 9159e9417d2Sbeveloper len = min_c(len, (int)sizeof(request.name) - 1); 9169e9417d2Sbeveloper memcpy(request.name, name, len); 9179e9417d2Sbeveloper request.name[len] = 0; 9189e9417d2Sbeveloper } 9199e9417d2Sbeveloper request.require_kinds = node_kinds; 9209e9417d2Sbeveloper 9219e9417d2Sbeveloper rv = QueryServer(SERVER_GET_LIVE_NODES, &request, sizeof(request), &reply, sizeof(reply)); 9229e9417d2Sbeveloper if (rv != B_OK) { 923353b9f6bSbeveloper FATAL("BMediaRoster::GetLiveNodes failed\n"); 9241299bfb2Sbeveloper *io_total_count = 0; 9259e9417d2Sbeveloper return rv; 9269e9417d2Sbeveloper } 9279e9417d2Sbeveloper 9289e9417d2Sbeveloper if (reply.count > MAX_LIVE_INFO) { 9299e9417d2Sbeveloper live_node_info *live_info; 9309e9417d2Sbeveloper area_id clone; 9319e9417d2Sbeveloper 9329e9417d2Sbeveloper clone = clone_area("live_node_info clone", reinterpret_cast<void **>(&live_info), B_ANY_ADDRESS, B_READ_AREA | B_WRITE_AREA, reply.area); 9339e9417d2Sbeveloper if (clone < B_OK) { 934353b9f6bSbeveloper FATAL("BMediaRoster::GetLiveNodes failed to clone area, %#lx\n", clone); 9359e9417d2Sbeveloper delete_area(reply.area); 9361299bfb2Sbeveloper *io_total_count = 0; 93752a38012Sejakowatz return B_ERROR; 93852a38012Sejakowatz } 93952a38012Sejakowatz 9409e9417d2Sbeveloper for (int32 i = 0; i < reply.count; i++) { 9419e9417d2Sbeveloper out_live_nodes[i] = live_info[i]; 9429e9417d2Sbeveloper } 9439e9417d2Sbeveloper 9449e9417d2Sbeveloper delete_area(clone); 9459e9417d2Sbeveloper delete_area(reply.area); 9469e9417d2Sbeveloper } else { 9479e9417d2Sbeveloper for (int32 i = 0; i < reply.count; i++) { 9489e9417d2Sbeveloper out_live_nodes[i] = reply.live_info[i]; 9499e9417d2Sbeveloper } 9509e9417d2Sbeveloper } 9519e9417d2Sbeveloper *io_total_count = reply.count; 9529e9417d2Sbeveloper 9539e9417d2Sbeveloper return B_OK; 9549e9417d2Sbeveloper } 9559e9417d2Sbeveloper 95652a38012Sejakowatz 95752a38012Sejakowatz status_t 95852a38012Sejakowatz BMediaRoster::GetFreeInputsFor(const media_node & node, 95952a38012Sejakowatz media_input * out_free_inputs, 96052a38012Sejakowatz int32 buf_num_inputs, 96152a38012Sejakowatz int32 * out_total_count, 96252a38012Sejakowatz media_type filter_type) 96352a38012Sejakowatz { 9643620737cSbeveloper CALLED(); 9659e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_CONSUMER) == 0) 9663620737cSbeveloper return B_MEDIA_BAD_NODE; 9673620737cSbeveloper if (out_free_inputs == NULL || out_total_count == NULL) 9683620737cSbeveloper return B_BAD_VALUE; 9693620737cSbeveloper 97076669a29Sbeveloper List<media_input> list; 9713620737cSbeveloper media_input *input; 9723620737cSbeveloper status_t rv; 9733620737cSbeveloper 9741299bfb2Sbeveloper *out_total_count = 0; 9751299bfb2Sbeveloper 97676669a29Sbeveloper rv = GetAllInputs(node, &list); 9773620737cSbeveloper if (B_OK != rv) 9783620737cSbeveloper return rv; 9793620737cSbeveloper 98076669a29Sbeveloper int32 i; 98176669a29Sbeveloper for (i = 0, list.Rewind(); list.GetNext(&input); i++) { 9823620737cSbeveloper if (filter_type != B_MEDIA_UNKNOWN_TYPE && filter_type != input->format.type) 9833620737cSbeveloper continue; // media_type used, but doesn't match 9843620737cSbeveloper if (input->source != media_source::null) 9853620737cSbeveloper continue; // consumer source already connected 9863620737cSbeveloper out_free_inputs[i] = *input; 9873620737cSbeveloper *out_total_count += 1; 9883620737cSbeveloper buf_num_inputs -= 1; 9893620737cSbeveloper if (buf_num_inputs == 0) 9903620737cSbeveloper break; 9913620737cSbeveloper } 9923620737cSbeveloper 99376669a29Sbeveloper PublishInputs(node, &list); 9943620737cSbeveloper return B_OK; 99552a38012Sejakowatz } 99652a38012Sejakowatz 99752a38012Sejakowatz 99852a38012Sejakowatz status_t 99952a38012Sejakowatz BMediaRoster::GetConnectedInputsFor(const media_node & node, 100052a38012Sejakowatz media_input * out_active_inputs, 100152a38012Sejakowatz int32 buf_num_inputs, 100252a38012Sejakowatz int32 * out_total_count) 100352a38012Sejakowatz { 10043620737cSbeveloper CALLED(); 10059e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_CONSUMER) == 0) 10063620737cSbeveloper return B_MEDIA_BAD_NODE; 10073620737cSbeveloper if (out_active_inputs == NULL || out_total_count == NULL) 10083620737cSbeveloper return B_BAD_VALUE; 10093620737cSbeveloper 101076669a29Sbeveloper List<media_input> list; 10113620737cSbeveloper media_input *input; 10123620737cSbeveloper status_t rv; 10133620737cSbeveloper 10141299bfb2Sbeveloper *out_total_count = 0; 10151299bfb2Sbeveloper 101676669a29Sbeveloper rv = GetAllInputs(node, &list); 10173620737cSbeveloper if (B_OK != rv) 10183620737cSbeveloper return rv; 10193620737cSbeveloper 102076669a29Sbeveloper int32 i; 102176669a29Sbeveloper for (i = 0, list.Rewind(); list.GetNext(&input); i++) { 10223620737cSbeveloper if (input->source == media_source::null) 10233620737cSbeveloper continue; // consumer source not connected 10243620737cSbeveloper out_active_inputs[i] = *input; 10253620737cSbeveloper *out_total_count += 1; 10263620737cSbeveloper buf_num_inputs -= 1; 10273620737cSbeveloper if (buf_num_inputs == 0) 10283620737cSbeveloper break; 10293620737cSbeveloper } 10303620737cSbeveloper 103176669a29Sbeveloper PublishInputs(node, &list); 10323620737cSbeveloper return B_OK; 103352a38012Sejakowatz } 103452a38012Sejakowatz 103552a38012Sejakowatz 103652a38012Sejakowatz status_t 103752a38012Sejakowatz BMediaRoster::GetAllInputsFor(const media_node & node, 103852a38012Sejakowatz media_input * out_inputs, 103952a38012Sejakowatz int32 buf_num_inputs, 104052a38012Sejakowatz int32 * out_total_count) 104152a38012Sejakowatz { 104252a38012Sejakowatz CALLED(); 10439e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_CONSUMER) == 0) 104452a38012Sejakowatz return B_MEDIA_BAD_NODE; 104552a38012Sejakowatz if (out_inputs == NULL || out_total_count == NULL) 104652a38012Sejakowatz return B_BAD_VALUE; 104752a38012Sejakowatz 104876669a29Sbeveloper List<media_input> list; 10493620737cSbeveloper media_input *input; 105052a38012Sejakowatz status_t rv; 105152a38012Sejakowatz 10521299bfb2Sbeveloper *out_total_count = 0; 10531299bfb2Sbeveloper 105476669a29Sbeveloper rv = GetAllInputs(node, &list); 10553620737cSbeveloper if (B_OK != rv) 10563620737cSbeveloper return rv; 10573620737cSbeveloper 105876669a29Sbeveloper int32 i; 105976669a29Sbeveloper for (i = 0, list.Rewind(); list.GetNext(&input); i++) { 10603620737cSbeveloper out_inputs[i] = *input; 106152a38012Sejakowatz *out_total_count += 1; 10623620737cSbeveloper buf_num_inputs -= 1; 10633620737cSbeveloper if (buf_num_inputs == 0) 10643620737cSbeveloper break; 106552a38012Sejakowatz } 106652a38012Sejakowatz 106776669a29Sbeveloper PublishInputs(node, &list); 10683620737cSbeveloper return B_OK; 106952a38012Sejakowatz } 107052a38012Sejakowatz 107152a38012Sejakowatz 107252a38012Sejakowatz status_t 107352a38012Sejakowatz BMediaRoster::GetFreeOutputsFor(const media_node & node, 107452a38012Sejakowatz media_output * out_free_outputs, 107552a38012Sejakowatz int32 buf_num_outputs, 107652a38012Sejakowatz int32 * out_total_count, 107752a38012Sejakowatz media_type filter_type) 107852a38012Sejakowatz { 10793620737cSbeveloper CALLED(); 10809e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_PRODUCER) == 0) 10813620737cSbeveloper return B_MEDIA_BAD_NODE; 10823620737cSbeveloper if (out_free_outputs == NULL || out_total_count == NULL) 10833620737cSbeveloper return B_BAD_VALUE; 10843620737cSbeveloper 108576669a29Sbeveloper List<media_output> list; 10863620737cSbeveloper media_output *output; 10873620737cSbeveloper status_t rv; 10883620737cSbeveloper 10891299bfb2Sbeveloper *out_total_count = 0; 10901299bfb2Sbeveloper 109176669a29Sbeveloper rv = GetAllOutputs(node, &list); 10923620737cSbeveloper if (B_OK != rv) 10933620737cSbeveloper return rv; 10943620737cSbeveloper 109576669a29Sbeveloper int32 i; 109676669a29Sbeveloper for (i = 0, list.Rewind(); list.GetNext(&output); i++) { 10973620737cSbeveloper if (filter_type != B_MEDIA_UNKNOWN_TYPE && filter_type != output->format.type) 10983620737cSbeveloper continue; // media_type used, but doesn't match 10993620737cSbeveloper if (output->destination != media_destination::null) 11003620737cSbeveloper continue; // producer destination already connected 11013620737cSbeveloper out_free_outputs[i] = *output; 11023620737cSbeveloper *out_total_count += 1; 11033620737cSbeveloper buf_num_outputs -= 1; 11043620737cSbeveloper if (buf_num_outputs == 0) 11053620737cSbeveloper break; 11063620737cSbeveloper } 11073620737cSbeveloper 110876669a29Sbeveloper PublishOutputs(node, &list); 11093620737cSbeveloper return B_OK; 111052a38012Sejakowatz } 111152a38012Sejakowatz 111252a38012Sejakowatz 111352a38012Sejakowatz status_t 111452a38012Sejakowatz BMediaRoster::GetConnectedOutputsFor(const media_node & node, 111552a38012Sejakowatz media_output * out_active_outputs, 111652a38012Sejakowatz int32 buf_num_outputs, 111752a38012Sejakowatz int32 * out_total_count) 111852a38012Sejakowatz { 11193620737cSbeveloper CALLED(); 11209e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_PRODUCER) == 0) 11213620737cSbeveloper return B_MEDIA_BAD_NODE; 11223620737cSbeveloper if (out_active_outputs == NULL || out_total_count == NULL) 11233620737cSbeveloper return B_BAD_VALUE; 11243620737cSbeveloper 112576669a29Sbeveloper List<media_output> list; 11263620737cSbeveloper media_output *output; 11273620737cSbeveloper status_t rv; 11283620737cSbeveloper 11291299bfb2Sbeveloper *out_total_count = 0; 11301299bfb2Sbeveloper 113176669a29Sbeveloper rv = GetAllOutputs(node, &list); 11323620737cSbeveloper if (B_OK != rv) 11333620737cSbeveloper return rv; 11343620737cSbeveloper 113576669a29Sbeveloper int32 i; 113676669a29Sbeveloper for (i = 0, list.Rewind(); list.GetNext(&output); i++) { 11373620737cSbeveloper if (output->destination == media_destination::null) 11383620737cSbeveloper continue; // producer destination not connected 11393620737cSbeveloper out_active_outputs[i] = *output; 11403620737cSbeveloper *out_total_count += 1; 11413620737cSbeveloper buf_num_outputs -= 1; 11423620737cSbeveloper if (buf_num_outputs == 0) 11433620737cSbeveloper break; 11443620737cSbeveloper } 11453620737cSbeveloper 114676669a29Sbeveloper PublishOutputs(node, &list); 11473620737cSbeveloper return B_OK; 114852a38012Sejakowatz } 114952a38012Sejakowatz 115052a38012Sejakowatz 115152a38012Sejakowatz status_t 115252a38012Sejakowatz BMediaRoster::GetAllOutputsFor(const media_node & node, 115352a38012Sejakowatz media_output * out_outputs, 115452a38012Sejakowatz int32 buf_num_outputs, 115552a38012Sejakowatz int32 * out_total_count) 115652a38012Sejakowatz { 115752a38012Sejakowatz CALLED(); 11589e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_PRODUCER) == 0) 115952a38012Sejakowatz return B_MEDIA_BAD_NODE; 116052a38012Sejakowatz if (out_outputs == NULL || out_total_count == NULL) 116152a38012Sejakowatz return B_BAD_VALUE; 116252a38012Sejakowatz 116376669a29Sbeveloper List<media_output> list; 11643620737cSbeveloper media_output *output; 116552a38012Sejakowatz status_t rv; 116652a38012Sejakowatz 11671299bfb2Sbeveloper *out_total_count = 0; 11681299bfb2Sbeveloper 116976669a29Sbeveloper rv = GetAllOutputs(node, &list); 11703620737cSbeveloper if (B_OK != rv) 11713620737cSbeveloper return rv; 11723620737cSbeveloper 117376669a29Sbeveloper int32 i; 117476669a29Sbeveloper for (i = 0, list.Rewind(); list.GetNext(&output); i++) { 11753620737cSbeveloper out_outputs[i] = *output; 117652a38012Sejakowatz *out_total_count += 1; 11773620737cSbeveloper buf_num_outputs -= 1; 11783620737cSbeveloper if (buf_num_outputs == 0) 11793620737cSbeveloper break; 118052a38012Sejakowatz } 118152a38012Sejakowatz 118276669a29Sbeveloper PublishOutputs(node, &list); 11833620737cSbeveloper return B_OK; 118452a38012Sejakowatz } 118552a38012Sejakowatz 118652a38012Sejakowatz 118752a38012Sejakowatz status_t 118852a38012Sejakowatz BMediaRoster::StartWatching(const BMessenger & where) 118952a38012Sejakowatz { 1190eae26d3dSbeveloper CALLED(); 1191eae26d3dSbeveloper if (!where.IsValid()) { 1192353b9f6bSbeveloper FATAL("BMediaRoster::StartWatching: messenger invalid!\n"); 1193eae26d3dSbeveloper return B_BAD_VALUE; 1194eae26d3dSbeveloper } 1195f4925104Sbeveloper return BPrivate::media::notifications::Register(where, media_node::null, B_MEDIA_WILDCARD); 119652a38012Sejakowatz } 119752a38012Sejakowatz 119852a38012Sejakowatz 119952a38012Sejakowatz status_t 120052a38012Sejakowatz BMediaRoster::StartWatching(const BMessenger & where, 120152a38012Sejakowatz int32 notificationType) 120252a38012Sejakowatz { 1203eae26d3dSbeveloper CALLED(); 1204eae26d3dSbeveloper if (!where.IsValid()) { 1205353b9f6bSbeveloper FATAL("BMediaRoster::StartWatching: messenger invalid!\n"); 1206eae26d3dSbeveloper return B_BAD_VALUE; 1207eae26d3dSbeveloper } 1208f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(false, notificationType)) { 1209353b9f6bSbeveloper FATAL("BMediaRoster::StartWatching: notificationType invalid!\n"); 1210eae26d3dSbeveloper return B_BAD_VALUE; 1211eae26d3dSbeveloper } 1212f4925104Sbeveloper return BPrivate::media::notifications::Register(where, media_node::null, notificationType); 121352a38012Sejakowatz } 121452a38012Sejakowatz 121552a38012Sejakowatz 121652a38012Sejakowatz status_t 121752a38012Sejakowatz BMediaRoster::StartWatching(const BMessenger & where, 121852a38012Sejakowatz const media_node & node, 121952a38012Sejakowatz int32 notificationType) 122052a38012Sejakowatz { 1221eae26d3dSbeveloper CALLED(); 1222eae26d3dSbeveloper if (!where.IsValid()) { 1223353b9f6bSbeveloper FATAL("BMediaRoster::StartWatching: messenger invalid!\n"); 1224eae26d3dSbeveloper return B_BAD_VALUE; 1225eae26d3dSbeveloper } 12269e9417d2Sbeveloper if (node.node <= 0) { 1227353b9f6bSbeveloper FATAL("BMediaRoster::StartWatching: node invalid!\n"); 1228eae26d3dSbeveloper return B_MEDIA_BAD_NODE; 1229eae26d3dSbeveloper } 1230f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(true, notificationType)) { 1231353b9f6bSbeveloper FATAL("BMediaRoster::StartWatching: notificationType invalid!\n"); 1232eae26d3dSbeveloper return B_BAD_VALUE; 1233eae26d3dSbeveloper } 1234f4925104Sbeveloper return BPrivate::media::notifications::Register(where, node, notificationType); 123552a38012Sejakowatz } 123652a38012Sejakowatz 123752a38012Sejakowatz 123852a38012Sejakowatz status_t 123952a38012Sejakowatz BMediaRoster::StopWatching(const BMessenger & where) 124052a38012Sejakowatz { 1241eae26d3dSbeveloper CALLED(); 1242eae26d3dSbeveloper // messenger may already be invalid, so we don't check this 1243f4925104Sbeveloper return BPrivate::media::notifications::Unregister(where, media_node::null, B_MEDIA_WILDCARD); 124452a38012Sejakowatz } 124552a38012Sejakowatz 124652a38012Sejakowatz 124752a38012Sejakowatz status_t 124852a38012Sejakowatz BMediaRoster::StopWatching(const BMessenger & where, 124952a38012Sejakowatz int32 notificationType) 125052a38012Sejakowatz { 1251eae26d3dSbeveloper CALLED(); 1252eae26d3dSbeveloper // messenger may already be invalid, so we don't check this 1253f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(false, notificationType)) { 1254353b9f6bSbeveloper FATAL("BMediaRoster::StopWatching: notificationType invalid!\n"); 1255eae26d3dSbeveloper return B_BAD_VALUE; 1256eae26d3dSbeveloper } 1257f4925104Sbeveloper return BPrivate::media::notifications::Unregister(where, media_node::null, notificationType); 125852a38012Sejakowatz } 125952a38012Sejakowatz 126052a38012Sejakowatz 126152a38012Sejakowatz status_t 126252a38012Sejakowatz BMediaRoster::StopWatching(const BMessenger & where, 126352a38012Sejakowatz const media_node & node, 126452a38012Sejakowatz int32 notificationType) 126552a38012Sejakowatz { 1266eae26d3dSbeveloper CALLED(); 1267eae26d3dSbeveloper // messenger may already be invalid, so we don't check this 12689e9417d2Sbeveloper if (node.node <= 0) { 1269353b9f6bSbeveloper FATAL("BMediaRoster::StopWatching: node invalid!\n"); 1270eae26d3dSbeveloper return B_MEDIA_BAD_NODE; 1271eae26d3dSbeveloper } 1272f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(true, notificationType)) { 1273353b9f6bSbeveloper FATAL("BMediaRoster::StopWatching: notificationType invalid!\n"); 1274eae26d3dSbeveloper return B_BAD_VALUE; 1275eae26d3dSbeveloper } 1276f4925104Sbeveloper return BPrivate::media::notifications::Unregister(where, node, notificationType); 127752a38012Sejakowatz } 127852a38012Sejakowatz 127952a38012Sejakowatz 128052a38012Sejakowatz status_t 128152a38012Sejakowatz BMediaRoster::RegisterNode(BMediaNode * node) 128252a38012Sejakowatz { 128352a38012Sejakowatz CALLED(); 128452a38012Sejakowatz if (node == NULL) 128552a38012Sejakowatz return B_BAD_VALUE; 128652a38012Sejakowatz 12873620737cSbeveloper status_t rv; 12889e9417d2Sbeveloper BMediaAddOn *addon; 12899e9417d2Sbeveloper int32 addon_flavor_id; 12909e9417d2Sbeveloper media_addon_id addon_id; 12913620737cSbeveloper 12929e9417d2Sbeveloper addon_flavor_id = 0; 12939e9417d2Sbeveloper addon = node->AddOn(&addon_flavor_id); 12949e9417d2Sbeveloper addon_id = addon ? addon->AddonID() : -1; 129552a38012Sejakowatz 12969e9417d2Sbeveloper server_register_node_request request; 12979e9417d2Sbeveloper server_register_node_reply reply; 12989e9417d2Sbeveloper 12999e9417d2Sbeveloper request.addon_id = addon_id; 13009e9417d2Sbeveloper request.addon_flavor_id = addon_flavor_id; 13019e9417d2Sbeveloper strcpy(request.name, node->Name()); 13029e9417d2Sbeveloper request.kinds = node->Kinds(); 13039e9417d2Sbeveloper request.port = node->ControlPort(); 13049e9417d2Sbeveloper request.team = team; 13059e9417d2Sbeveloper 13061299bfb2Sbeveloper TRACE("BMediaRoster::RegisterNode: sending SERVER_REGISTER_NODE: port %ld, kinds %#Lx, team %ld, name '%s'\n", request.port, request.kinds, request.team, request.name); 13071299bfb2Sbeveloper 13089e9417d2Sbeveloper rv = QueryServer(SERVER_REGISTER_NODE, &request, sizeof(request), &reply, sizeof(reply)); 13099e9417d2Sbeveloper if (rv != B_OK) { 1310353b9f6bSbeveloper FATAL("BMediaRoster::RegisterNode: failed to register node %s (error %#lx)\n", node->Name(), rv); 13119e9417d2Sbeveloper return rv; 13129e9417d2Sbeveloper } 13139e9417d2Sbeveloper 13149e9417d2Sbeveloper // we are a friend class of BMediaNode and initilize this member variable 13159e9417d2Sbeveloper node->fNodeID = reply.nodeid; 13169e9417d2Sbeveloper ASSERT(reply.nodeid == node->Node().node); 13179e9417d2Sbeveloper ASSERT(reply.nodeid == node->ID()); 13189e9417d2Sbeveloper 13199e9417d2Sbeveloper // call the callback 13209e9417d2Sbeveloper node->NodeRegistered(); 13213620737cSbeveloper 1322570f7d04Sbeveloper /* 13233620737cSbeveloper // register existing inputs and outputs with the 13243620737cSbeveloper // media_server, this allows GetLiveNodes() to work 13253620737cSbeveloper // with created, but unconnected nodes. 13263620737cSbeveloper if (node->Kinds() & B_BUFFER_PRODUCER) { 13273620737cSbeveloper Stack<media_output> stack; 13283620737cSbeveloper if (B_OK == GetAllOutputs(node->Node(), &stack)) 13293620737cSbeveloper PublishOutputs(node->Node(), &stack); 13303620737cSbeveloper } else if (node->Kinds() & B_BUFFER_CONSUMER) { 13313620737cSbeveloper Stack<media_input> stack; 13323620737cSbeveloper if (B_OK == GetAllInputs(node->Node(), &stack)) 13333620737cSbeveloper PublishInputs(node->Node(), &stack); 13343620737cSbeveloper } 1335570f7d04Sbeveloper */ 1336570f7d04Sbeveloper 13371299bfb2Sbeveloper BPrivate::media::notifications::NodesCreated(&reply.nodeid, 1); 1338570f7d04Sbeveloper /* 1339570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: registered node name '%s', id %ld, addon %ld, flavor %ld\n", node->Name(), node->ID(), addon_id, addon_flavor_id); 1340570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: node this %p\n", node); 1341570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: node fConsumerThis %p\n", node->fConsumerThis); 1342570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: node fProducerThis %p\n", node->fProducerThis); 1343570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: node fFileInterfaceThis %p\n", node->fFileInterfaceThis); 1344570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: node fControllableThis %p\n", node->fControllableThis); 1345570f7d04Sbeveloper TRACE("BMediaRoster::RegisterNode: node fTimeSourceThis %p\n", node->fTimeSourceThis); 1346570f7d04Sbeveloper */ 13479e9417d2Sbeveloper 13489e9417d2Sbeveloper return B_OK; 134952a38012Sejakowatz } 135052a38012Sejakowatz 135152a38012Sejakowatz 135252a38012Sejakowatz status_t 135352a38012Sejakowatz BMediaRoster::UnregisterNode(BMediaNode * node) 135452a38012Sejakowatz { 13559e9417d2Sbeveloper CALLED(); 13569e9417d2Sbeveloper if (node == NULL) 13579e9417d2Sbeveloper return B_BAD_VALUE; 13589e9417d2Sbeveloper 13599e9417d2Sbeveloper if (node->fRefCount != 0) { 1360353b9f6bSbeveloper FATAL("BMediaRoster::UnregisterNode: Warning node name '%s' has local reference count of %ld\n", node->Name(), node->fRefCount); 13619e9417d2Sbeveloper // no return here, we continue and unregister! 13629e9417d2Sbeveloper } 13639e9417d2Sbeveloper if (node->ID() == -2) { 1364353b9f6bSbeveloper FATAL("BMediaRoster::UnregisterNode: Warning node name '%s' already unregistered\n", node->Name()); 13659e9417d2Sbeveloper return B_OK; 13669e9417d2Sbeveloper } 13679e9417d2Sbeveloper 13689e9417d2Sbeveloper server_unregister_node_request request; 13699e9417d2Sbeveloper server_unregister_node_reply reply; 13709e9417d2Sbeveloper status_t rv; 13719e9417d2Sbeveloper 13729e9417d2Sbeveloper request.nodeid = node->ID(); 13739e9417d2Sbeveloper request.team = team; 13749e9417d2Sbeveloper 13751299bfb2Sbeveloper // send a notification 13761299bfb2Sbeveloper BPrivate::media::notifications::NodesDeleted(&request.nodeid, 1); 13771299bfb2Sbeveloper 13789e9417d2Sbeveloper rv = QueryServer(SERVER_UNREGISTER_NODE, &request, sizeof(request), &reply, sizeof(reply)); 13799e9417d2Sbeveloper if (rv != B_OK) { 1380353b9f6bSbeveloper FATAL("BMediaRoster::UnregisterNode: failed to unregister node name '%s' (error %#lx)\n", node->Name(), rv); 13819e9417d2Sbeveloper return rv; 13829e9417d2Sbeveloper } 13839e9417d2Sbeveloper 13849e9417d2Sbeveloper if (reply.addon_id != -1) 13859e9417d2Sbeveloper _DormantNodeManager->PutAddon(reply.addon_id); 13869e9417d2Sbeveloper 13871299bfb2Sbeveloper // we are a friend class of BMediaNode and invalidate this member variable 13889e9417d2Sbeveloper node->fNodeID = -2; 13899e9417d2Sbeveloper 13909e9417d2Sbeveloper return B_OK; 139152a38012Sejakowatz } 139252a38012Sejakowatz 139352a38012Sejakowatz 139452a38012Sejakowatz // thread safe for multiple calls to Roster() 139552a38012Sejakowatz /* static */ BMediaRoster * 139652a38012Sejakowatz BMediaRoster::Roster(status_t* out_error) 139752a38012Sejakowatz { 139852a38012Sejakowatz static BLocker locker("BMediaRoster::Roster locker"); 139952a38012Sejakowatz locker.Lock(); 140052a38012Sejakowatz if (_sDefault == NULL) { 140152a38012Sejakowatz _sDefault = new BMediaRoster(); 140252a38012Sejakowatz if (out_error != NULL) 140352a38012Sejakowatz *out_error = B_OK; 140452a38012Sejakowatz } else { 140552a38012Sejakowatz if (out_error != NULL) 140652a38012Sejakowatz *out_error = B_OK; 140752a38012Sejakowatz } 140852a38012Sejakowatz locker.Unlock(); 140952a38012Sejakowatz return _sDefault; 141052a38012Sejakowatz } 141152a38012Sejakowatz 141252a38012Sejakowatz 141352a38012Sejakowatz // won't create it if there isn't one 141452a38012Sejakowatz // not thread safe if you call Roster() at the same time 141552a38012Sejakowatz /* static */ BMediaRoster * 141652a38012Sejakowatz BMediaRoster::CurrentRoster() 141752a38012Sejakowatz { 141852a38012Sejakowatz return _sDefault; 141952a38012Sejakowatz } 142052a38012Sejakowatz 142152a38012Sejakowatz 142252a38012Sejakowatz status_t 142352a38012Sejakowatz BMediaRoster::SetTimeSourceFor(media_node_id node, 142452a38012Sejakowatz media_node_id time_source) 142552a38012Sejakowatz { 142652a38012Sejakowatz UNIMPLEMENTED(); 142752a38012Sejakowatz return B_ERROR; 142852a38012Sejakowatz } 142952a38012Sejakowatz 143052a38012Sejakowatz 143152a38012Sejakowatz status_t 143252a38012Sejakowatz BMediaRoster::GetParameterWebFor(const media_node & node, 143352a38012Sejakowatz BParameterWeb ** out_web) 143452a38012Sejakowatz { 143552a38012Sejakowatz UNIMPLEMENTED(); 143652a38012Sejakowatz return B_ERROR; 143752a38012Sejakowatz } 143852a38012Sejakowatz 143952a38012Sejakowatz 144052a38012Sejakowatz status_t 144152a38012Sejakowatz BMediaRoster::StartControlPanel(const media_node & node, 144252a38012Sejakowatz BMessenger * out_messenger) 144352a38012Sejakowatz { 144452a38012Sejakowatz UNIMPLEMENTED(); 144552a38012Sejakowatz return B_ERROR; 144652a38012Sejakowatz } 144752a38012Sejakowatz 144852a38012Sejakowatz 144952a38012Sejakowatz status_t 145052a38012Sejakowatz BMediaRoster::GetDormantNodes(dormant_node_info * out_info, 145152a38012Sejakowatz int32 * io_count, 145252a38012Sejakowatz const media_format * has_input /* = NULL */, 145352a38012Sejakowatz const media_format * has_output /* = NULL */, 145452a38012Sejakowatz const char * name /* = NULL */, 145552a38012Sejakowatz uint64 require_kinds /* = NULL */, 145652a38012Sejakowatz uint64 deny_kinds /* = NULL */) 145752a38012Sejakowatz { 145852a38012Sejakowatz CALLED(); 145952a38012Sejakowatz if (out_info == NULL) 146052a38012Sejakowatz return B_BAD_VALUE; 146152a38012Sejakowatz if (io_count == NULL) 146252a38012Sejakowatz return B_BAD_VALUE; 146352a38012Sejakowatz if (*io_count <= 0) 146452a38012Sejakowatz return B_BAD_VALUE; 146552a38012Sejakowatz 146652a38012Sejakowatz xfer_server_get_dormant_nodes msg; 146752a38012Sejakowatz port_id port; 146852a38012Sejakowatz status_t rv; 146952a38012Sejakowatz 147052a38012Sejakowatz port = find_port("media_server port"); 147152a38012Sejakowatz if (port <= B_OK) 147252a38012Sejakowatz return B_ERROR; 147352a38012Sejakowatz 147452a38012Sejakowatz msg.maxcount = *io_count; 147552a38012Sejakowatz msg.has_input = (bool) has_input; 147652a38012Sejakowatz if (has_input) 147752a38012Sejakowatz msg.inputformat = *has_input; // XXX we should not make a flat copy of media_format 147852a38012Sejakowatz msg.has_output = (bool) has_output; 147952a38012Sejakowatz if (has_output) 148052a38012Sejakowatz msg.outputformat = *has_output;; // XXX we should not make a flat copy of media_format 148152a38012Sejakowatz msg.has_name = (bool) name; 148252a38012Sejakowatz if (name) { 14839e9417d2Sbeveloper int len = strlen(name); 14849e9417d2Sbeveloper len = min_c(len, (int)sizeof(msg.name) - 1); 148552a38012Sejakowatz memcpy(msg.name, name, len); 148652a38012Sejakowatz msg.name[len] = 0; 148752a38012Sejakowatz } 148852a38012Sejakowatz msg.require_kinds = require_kinds; 148952a38012Sejakowatz msg.deny_kinds = deny_kinds; 149052a38012Sejakowatz msg.reply_port = _PortPool->GetPort(); 149152a38012Sejakowatz 149252a38012Sejakowatz rv = write_port(port, SERVER_GET_DORMANT_NODES, &msg, sizeof(msg)); 149352a38012Sejakowatz if (rv != B_OK) { 149452a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 149552a38012Sejakowatz return rv; 149652a38012Sejakowatz } 149752a38012Sejakowatz 149852a38012Sejakowatz xfer_server_get_dormant_nodes_reply reply; 149952a38012Sejakowatz int32 code; 150052a38012Sejakowatz 150152a38012Sejakowatz rv = read_port(msg.reply_port, &code, &reply, sizeof(reply)); 150252a38012Sejakowatz if (rv < B_OK) { 150352a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 150452a38012Sejakowatz return rv; 150552a38012Sejakowatz } 150652a38012Sejakowatz 150752a38012Sejakowatz *io_count = reply.count; 150852a38012Sejakowatz 150952a38012Sejakowatz if (*io_count > 0) { 151052a38012Sejakowatz rv = read_port(msg.reply_port, &code, out_info, *io_count * sizeof(dormant_node_info)); 151152a38012Sejakowatz if (rv < B_OK) 151252a38012Sejakowatz reply.result = rv; 151352a38012Sejakowatz } 151452a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 151552a38012Sejakowatz 151652a38012Sejakowatz return reply.result; 151752a38012Sejakowatz } 151852a38012Sejakowatz 151952a38012Sejakowatz 152052a38012Sejakowatz status_t 152152a38012Sejakowatz BMediaRoster::InstantiateDormantNode(const dormant_node_info & in_info, 152252a38012Sejakowatz media_node * out_node, 152352a38012Sejakowatz uint32 flags /* currently B_FLAVOR_IS_GLOBAL or B_FLAVOR_IS_LOCAL */ ) 152452a38012Sejakowatz { 152552a38012Sejakowatz CALLED(); 152652a38012Sejakowatz if ((flags & (B_FLAVOR_IS_GLOBAL | B_FLAVOR_IS_LOCAL)) == 0) { 1527*618b1338Sbeveloper FATAL("Error: BMediaRoster::InstantiateDormantNode called without valid flags\n"); 152852a38012Sejakowatz return B_BAD_VALUE; 152952a38012Sejakowatz } 153052a38012Sejakowatz if (out_node == 0) 153152a38012Sejakowatz return B_BAD_VALUE; 153252a38012Sejakowatz 153352a38012Sejakowatz // XXX we should not trust the values passed in by the user, 153452a38012Sejakowatz // XXX and ask the server to determine where to insta 153552a38012Sejakowatz 1536e0207628Sbeveloper 1537e0207628Sbeveloper // XXX SOMETHING IS VERY WRONG HERE 1538e0207628Sbeveloper // if ((in_info.flavor_flags & B_FLAVOR_IS_GLOBAL) == 0 && (flags & B_FLAVOR_IS_LOCAL)) { 1539e0207628Sbeveloper if (flags & B_FLAVOR_IS_LOCAL) { 154052a38012Sejakowatz return InstantiateDormantNode(in_info,out_node); 154152a38012Sejakowatz } 154252a38012Sejakowatz 1543e0207628Sbeveloper // XXX SOMETHING IS VERY WRONG HERE 1544e0207628Sbeveloper // if ((in_info.flavor_flags & B_FLAVOR_IS_GLOBAL) || (flags & B_FLAVOR_IS_GLOBAL)) { 1545e0207628Sbeveloper if (flags & B_FLAVOR_IS_GLOBAL) { 154652a38012Sejakowatz // forward this request into the media_addon_server, 154752a38012Sejakowatz // which in turn will call InstantiateDormantNode() 154852a38012Sejakowatz // to create it there localy 15498c6a6096Sbeveloper addonserver_instantiate_dormant_node_request request; 15508c6a6096Sbeveloper addonserver_instantiate_dormant_node_reply reply; 155152a38012Sejakowatz status_t rv; 1552f4925104Sbeveloper 1553f4925104Sbeveloper request.info = in_info; 1554f4925104Sbeveloper rv = QueryAddonServer(ADDONSERVER_INSTANTIATE_DORMANT_NODE, &request, sizeof(request), &reply, sizeof(reply)); 1555f4925104Sbeveloper if (rv == B_OK) { 155652a38012Sejakowatz *out_node = reply.node; 1557f4925104Sbeveloper } 1558f4925104Sbeveloper return rv; 155952a38012Sejakowatz } 156052a38012Sejakowatz 1561e0207628Sbeveloper // XXX SOMETHING IS VERY WRONG HERE 1562*618b1338Sbeveloper FATAL("Error: BMediaRoster::InstantiateDormantNode addon_id %d, flavor_id %d, flags %#08lx\n", (int)in_info.addon, (int)in_info.flavor_id, flags); 156352a38012Sejakowatz 156452a38012Sejakowatz return B_ERROR; 156552a38012Sejakowatz } 156652a38012Sejakowatz 156752a38012Sejakowatz 156852a38012Sejakowatz status_t 156952a38012Sejakowatz BMediaRoster::InstantiateDormantNode(const dormant_node_info & in_info, 157052a38012Sejakowatz media_node * out_node) 157152a38012Sejakowatz { 157216b7eea4Sbeveloper CALLED(); 157352a38012Sejakowatz 157452a38012Sejakowatz // to instantiate a dormant node in the current address space, we need to 157552a38012Sejakowatz // either load the add-on from file and create a new BMediaAddOn class, or 157652a38012Sejakowatz // reuse the cached BMediaAddOn from a previous call 157752a38012Sejakowatz // call BMediaAddOn::InstantiateNodeFor() 157852a38012Sejakowatz // and cache the BMediaAddOn after that for later reuse. 157952a38012Sejakowatz // BeOS R5 does not seem to delete it when the application quits 158052a38012Sejakowatz // if B_FLAVOR_IS_GLOBAL, we need to use the BMediaAddOn object that 158152a38012Sejakowatz // resides in the media_addon_server 158252a38012Sejakowatz 158352a38012Sejakowatz // RegisterNode() is called automatically for nodes instantiated from add-ons 158452a38012Sejakowatz 158516b7eea4Sbeveloper //XXX TEST! 158616b7eea4Sbeveloper BMediaAddOn *addon; 158716b7eea4Sbeveloper BMediaNode *node; 158816b7eea4Sbeveloper BMessage config; 158916b7eea4Sbeveloper status_t out_error; 15908c6a6096Sbeveloper status_t rv; 159116b7eea4Sbeveloper addon = _DormantNodeManager->GetAddon(in_info.addon); 159216b7eea4Sbeveloper if (!addon) { 1593*618b1338Sbeveloper FATAL("BMediaRoster::InstantiateDormantNode: GetAddon failed\n"); 159452a38012Sejakowatz return B_ERROR; 159552a38012Sejakowatz } 15961299bfb2Sbeveloper flavor_info temp; // XXX fix this! 15971299bfb2Sbeveloper temp.name = "XXX flavor_info name"; 15981299bfb2Sbeveloper temp.info = "XXX flavor_info info"; 159916b7eea4Sbeveloper temp.internal_id = in_info.flavor_id; 160016b7eea4Sbeveloper node = addon->InstantiateNodeFor(&temp, &config, &out_error); 160116b7eea4Sbeveloper if (!node) { 1602*618b1338Sbeveloper FATAL("BMediaRoster::InstantiateDormantNode: InstantiateNodeFor failed\n"); 160316b7eea4Sbeveloper _DormantNodeManager->PutAddon(in_info.addon); 160416b7eea4Sbeveloper return B_ERROR; 160516b7eea4Sbeveloper } 16068c6a6096Sbeveloper rv = RegisterNode(node); 16078c6a6096Sbeveloper if (rv != B_OK) { 1608*618b1338Sbeveloper FATAL("BMediaRoster::InstantiateDormantNode: RegisterNode failed\n"); 16098c6a6096Sbeveloper delete node; 16108c6a6096Sbeveloper _DormantNodeManager->PutAddon(in_info.addon); 16118c6a6096Sbeveloper return B_ERROR; 16128c6a6096Sbeveloper } 16138c6a6096Sbeveloper 16148c6a6096Sbeveloper // XXX we must remember in_info.addon and call 16158c6a6096Sbeveloper // XXX _DormantNodeManager->PutAddon when the 16168c6a6096Sbeveloper // XXX node is unregistered 16179e9417d2Sbeveloper // should be handled by RegisterNode() and UnegisterNode() now 16188c6a6096Sbeveloper 161916b7eea4Sbeveloper *out_node = node->Node(); 162016b7eea4Sbeveloper return B_OK; 162116b7eea4Sbeveloper } 162252a38012Sejakowatz 162352a38012Sejakowatz 162452a38012Sejakowatz status_t 162552a38012Sejakowatz BMediaRoster::GetDormantNodeFor(const media_node & node, 162652a38012Sejakowatz dormant_node_info * out_info) 162752a38012Sejakowatz { 162885dfab2bSbeveloper CALLED(); 162985dfab2bSbeveloper if (out_info == NULL) 163085dfab2bSbeveloper return B_BAD_VALUE; 163185dfab2bSbeveloper if (node.node <= 0) 163285dfab2bSbeveloper return B_MEDIA_BAD_NODE; 163352a38012Sejakowatz 163485dfab2bSbeveloper server_get_dormant_node_for_request request; 163585dfab2bSbeveloper server_get_dormant_node_for_reply reply; 163685dfab2bSbeveloper status_t rv; 163785dfab2bSbeveloper 163885dfab2bSbeveloper request.node = node; 163985dfab2bSbeveloper 16401299bfb2Sbeveloper rv = QueryServer(SERVER_GET_DORMANT_NODE_FOR, &request, sizeof(request), &reply, sizeof(reply)); 164185dfab2bSbeveloper if (rv != B_OK) 164285dfab2bSbeveloper return rv; 164385dfab2bSbeveloper 164485dfab2bSbeveloper *out_info = reply.node_info; 164585dfab2bSbeveloper return B_OK; 164652a38012Sejakowatz } 164752a38012Sejakowatz 164852a38012Sejakowatz 164952a38012Sejakowatz status_t 165052a38012Sejakowatz BMediaRoster::GetDormantFlavorInfoFor(const dormant_node_info & in_dormant, 165152a38012Sejakowatz dormant_flavor_info * out_flavor) 165252a38012Sejakowatz { 165352a38012Sejakowatz CALLED(); 165452a38012Sejakowatz 165552a38012Sejakowatz xfer_server_get_dormant_flavor_info msg; 165652a38012Sejakowatz xfer_server_get_dormant_flavor_info_reply *reply; 165752a38012Sejakowatz port_id port; 165852a38012Sejakowatz status_t rv; 165952a38012Sejakowatz int32 code; 166052a38012Sejakowatz 166152a38012Sejakowatz port = find_port("media_server port"); 166252a38012Sejakowatz if (port <= B_OK) 166352a38012Sejakowatz return B_ERROR; 166452a38012Sejakowatz 166552a38012Sejakowatz reply = (xfer_server_get_dormant_flavor_info_reply *) malloc(16000); 166652a38012Sejakowatz if (reply == 0) 166752a38012Sejakowatz return B_ERROR; 166852a38012Sejakowatz 166952a38012Sejakowatz msg.addon = in_dormant.addon; 167052a38012Sejakowatz msg.flavor_id = in_dormant.flavor_id; 167152a38012Sejakowatz msg.reply_port = _PortPool->GetPort(); 167252a38012Sejakowatz rv = write_port(port, SERVER_GET_DORMANT_FLAVOR_INFO, &msg, sizeof(msg)); 167352a38012Sejakowatz if (rv != B_OK) { 167452a38012Sejakowatz free(reply); 167552a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 167652a38012Sejakowatz return rv; 167752a38012Sejakowatz } 167852a38012Sejakowatz rv = read_port(msg.reply_port, &code, reply, 16000); 167952a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 168052a38012Sejakowatz 168152a38012Sejakowatz if (rv < B_OK) { 168252a38012Sejakowatz free(reply); 168352a38012Sejakowatz return rv; 168452a38012Sejakowatz } 168552a38012Sejakowatz 168652a38012Sejakowatz if (reply->result == B_OK) 168752a38012Sejakowatz rv = out_flavor->Unflatten(reply->dfi_type, &reply->dfi, reply->dfi_size); 168852a38012Sejakowatz else 168952a38012Sejakowatz rv = reply->result; 169052a38012Sejakowatz 169152a38012Sejakowatz free(reply); 169252a38012Sejakowatz return rv; 169352a38012Sejakowatz } 169452a38012Sejakowatz 169552a38012Sejakowatz 169652a38012Sejakowatz status_t 169752a38012Sejakowatz BMediaRoster::GetLatencyFor(const media_node & producer, 169852a38012Sejakowatz bigtime_t * out_latency) 169952a38012Sejakowatz { 170052a38012Sejakowatz UNIMPLEMENTED(); 170152a38012Sejakowatz *out_latency = 0; 170252a38012Sejakowatz return B_ERROR; 170352a38012Sejakowatz } 170452a38012Sejakowatz 170552a38012Sejakowatz 170652a38012Sejakowatz status_t 170752a38012Sejakowatz BMediaRoster::GetInitialLatencyFor(const media_node & producer, 170852a38012Sejakowatz bigtime_t * out_latency, 170952a38012Sejakowatz uint32 * out_flags) 171052a38012Sejakowatz { 171152a38012Sejakowatz UNIMPLEMENTED(); 171252a38012Sejakowatz *out_latency = 0; 171352a38012Sejakowatz *out_flags = 0; 171452a38012Sejakowatz return B_ERROR; 171552a38012Sejakowatz } 171652a38012Sejakowatz 171752a38012Sejakowatz 171852a38012Sejakowatz status_t 171952a38012Sejakowatz BMediaRoster::GetStartLatencyFor(const media_node & time_source, 172052a38012Sejakowatz bigtime_t * out_latency) 172152a38012Sejakowatz { 172252a38012Sejakowatz UNIMPLEMENTED(); 172352a38012Sejakowatz *out_latency = 0; 172452a38012Sejakowatz return B_ERROR; 172552a38012Sejakowatz } 172652a38012Sejakowatz 172752a38012Sejakowatz 172852a38012Sejakowatz status_t 172952a38012Sejakowatz BMediaRoster::GetFileFormatsFor(const media_node & file_interface, 173052a38012Sejakowatz media_file_format * out_formats, 173152a38012Sejakowatz int32 * io_num_infos) 173252a38012Sejakowatz { 173352a38012Sejakowatz UNIMPLEMENTED(); 173452a38012Sejakowatz return B_ERROR; 173552a38012Sejakowatz } 173652a38012Sejakowatz 173752a38012Sejakowatz 173852a38012Sejakowatz status_t 173952a38012Sejakowatz BMediaRoster::SetRefFor(const media_node & file_interface, 174052a38012Sejakowatz const entry_ref & file, 174152a38012Sejakowatz bool create_and_truncate, 174252a38012Sejakowatz bigtime_t * out_length) /* if create is false */ 174352a38012Sejakowatz { 174452a38012Sejakowatz UNIMPLEMENTED(); 174552a38012Sejakowatz return B_ERROR; 174652a38012Sejakowatz } 174752a38012Sejakowatz 174852a38012Sejakowatz 174952a38012Sejakowatz status_t 175052a38012Sejakowatz BMediaRoster::GetRefFor(const media_node & node, 175152a38012Sejakowatz entry_ref * out_file, 175252a38012Sejakowatz BMimeType * mime_type) 175352a38012Sejakowatz { 175452a38012Sejakowatz UNIMPLEMENTED(); 175552a38012Sejakowatz return B_ERROR; 175652a38012Sejakowatz } 175752a38012Sejakowatz 175852a38012Sejakowatz 175952a38012Sejakowatz status_t 176052a38012Sejakowatz BMediaRoster::SniffRefFor(const media_node & file_interface, 176152a38012Sejakowatz const entry_ref & file, 176252a38012Sejakowatz BMimeType * mime_type, 176352a38012Sejakowatz float * out_capability) 176452a38012Sejakowatz { 176552a38012Sejakowatz UNIMPLEMENTED(); 176652a38012Sejakowatz return B_ERROR; 176752a38012Sejakowatz } 176852a38012Sejakowatz 176952a38012Sejakowatz 177052a38012Sejakowatz /* This is the generic "here's a file, now can someone please play it" interface */ 177152a38012Sejakowatz status_t 177252a38012Sejakowatz BMediaRoster::SniffRef(const entry_ref & file, 177352a38012Sejakowatz uint64 require_node_kinds, /* if you need an EntityInterface or BufferConsumer or something */ 177452a38012Sejakowatz dormant_node_info * out_node, 177552a38012Sejakowatz BMimeType * mime_type) 177652a38012Sejakowatz { 177752a38012Sejakowatz UNIMPLEMENTED(); 177852a38012Sejakowatz return B_ERROR; 177952a38012Sejakowatz } 178052a38012Sejakowatz 178152a38012Sejakowatz 178252a38012Sejakowatz status_t 178352a38012Sejakowatz BMediaRoster::GetDormantNodeForType(const BMimeType & type, 178452a38012Sejakowatz uint64 require_node_kinds, 178552a38012Sejakowatz dormant_node_info * out_node) 178652a38012Sejakowatz { 178752a38012Sejakowatz UNIMPLEMENTED(); 178852a38012Sejakowatz return B_ERROR; 178952a38012Sejakowatz } 179052a38012Sejakowatz 179152a38012Sejakowatz 179252a38012Sejakowatz status_t 179352a38012Sejakowatz BMediaRoster::GetReadFileFormatsFor(const dormant_node_info & in_node, 179452a38012Sejakowatz media_file_format * out_read_formats, 179552a38012Sejakowatz int32 in_read_count, 179652a38012Sejakowatz int32 * out_read_count) 179752a38012Sejakowatz { 179852a38012Sejakowatz UNIMPLEMENTED(); 179952a38012Sejakowatz return B_ERROR; 180052a38012Sejakowatz } 180152a38012Sejakowatz 180252a38012Sejakowatz 180352a38012Sejakowatz status_t 180452a38012Sejakowatz BMediaRoster::GetWriteFileFormatsFor(const dormant_node_info & in_node, 180552a38012Sejakowatz media_file_format * out_write_formats, 180652a38012Sejakowatz int32 in_write_count, 180752a38012Sejakowatz int32 * out_write_count) 180852a38012Sejakowatz { 180952a38012Sejakowatz UNIMPLEMENTED(); 181052a38012Sejakowatz return B_ERROR; 181152a38012Sejakowatz } 181252a38012Sejakowatz 181352a38012Sejakowatz 181452a38012Sejakowatz status_t 181552a38012Sejakowatz BMediaRoster::GetFormatFor(const media_output & output, 181652a38012Sejakowatz media_format * io_format, 181752a38012Sejakowatz uint32 flags) 181852a38012Sejakowatz { 181952a38012Sejakowatz UNIMPLEMENTED(); 182052a38012Sejakowatz return B_ERROR; 182152a38012Sejakowatz } 182252a38012Sejakowatz 182352a38012Sejakowatz 182452a38012Sejakowatz status_t 182552a38012Sejakowatz BMediaRoster::GetFormatFor(const media_input & input, 182652a38012Sejakowatz media_format * io_format, 182752a38012Sejakowatz uint32 flags) 182852a38012Sejakowatz { 182952a38012Sejakowatz UNIMPLEMENTED(); 183052a38012Sejakowatz return B_ERROR; 183152a38012Sejakowatz } 183252a38012Sejakowatz 183352a38012Sejakowatz 183452a38012Sejakowatz status_t 183552a38012Sejakowatz BMediaRoster::GetFormatFor(const media_node & node, 183652a38012Sejakowatz media_format * io_format, 183752a38012Sejakowatz float quality) 183852a38012Sejakowatz { 183952a38012Sejakowatz UNIMPLEMENTED(); 184052a38012Sejakowatz return B_ERROR; 184152a38012Sejakowatz } 184252a38012Sejakowatz 184352a38012Sejakowatz 184452a38012Sejakowatz ssize_t 184552a38012Sejakowatz BMediaRoster::GetNodeAttributesFor(const media_node & node, 184652a38012Sejakowatz media_node_attribute * outArray, 184752a38012Sejakowatz size_t inMaxCount) 184852a38012Sejakowatz { 184952a38012Sejakowatz UNIMPLEMENTED(); 185052a38012Sejakowatz return B_ERROR; 185152a38012Sejakowatz } 185252a38012Sejakowatz 185352a38012Sejakowatz 185452a38012Sejakowatz media_node_id 185552a38012Sejakowatz BMediaRoster::NodeIDFor(port_id source_or_destination_port) 185652a38012Sejakowatz { 18579e9417d2Sbeveloper CALLED(); 18589e9417d2Sbeveloper 18599e9417d2Sbeveloper server_node_id_for_request request; 18609e9417d2Sbeveloper server_node_id_for_reply reply; 18619e9417d2Sbeveloper status_t rv; 18629e9417d2Sbeveloper 18639e9417d2Sbeveloper request.port = source_or_destination_port; 18649e9417d2Sbeveloper 18659e9417d2Sbeveloper rv = QueryServer(SERVER_NODE_ID_FOR, &request, sizeof(request), &reply, sizeof(reply)); 18669e9417d2Sbeveloper if (rv != B_OK) { 1867353b9f6bSbeveloper FATAL("BMediaRoster::NodeIDFor: failed (error %#lx)\n", rv); 18689e9417d2Sbeveloper return -1; 18699e9417d2Sbeveloper } 18709e9417d2Sbeveloper 18719e9417d2Sbeveloper return reply.nodeid; 187252a38012Sejakowatz } 187352a38012Sejakowatz 187452a38012Sejakowatz 187552a38012Sejakowatz status_t 187652a38012Sejakowatz BMediaRoster::GetInstancesFor(media_addon_id addon, 187752a38012Sejakowatz int32 flavor, 187852a38012Sejakowatz media_node_id * out_id, 187952a38012Sejakowatz int32 * io_count) 188052a38012Sejakowatz { 188185dfab2bSbeveloper CALLED(); 188285dfab2bSbeveloper if (out_id == NULL || io_count == NULL) 188385dfab2bSbeveloper return B_BAD_VALUE; 188485dfab2bSbeveloper if (*io_count <= 0) 188585dfab2bSbeveloper return B_BAD_VALUE; 188685dfab2bSbeveloper 188785dfab2bSbeveloper server_get_instances_for_request request; 188885dfab2bSbeveloper server_get_instances_for_reply reply; 188985dfab2bSbeveloper status_t rv; 189085dfab2bSbeveloper 189185dfab2bSbeveloper request.maxcount = *io_count; 189285dfab2bSbeveloper request.addon_id = addon; 189385dfab2bSbeveloper request.addon_flavor_id = flavor; 189485dfab2bSbeveloper 189585dfab2bSbeveloper rv = QueryServer(SERVER_GET_INSTANCES_FOR, &request, sizeof(request), &reply, sizeof(reply)); 189685dfab2bSbeveloper if (rv != B_OK) { 1897353b9f6bSbeveloper FATAL("BMediaRoster::GetLiveNodes failed\n"); 189885dfab2bSbeveloper return rv; 189952a38012Sejakowatz } 190052a38012Sejakowatz 190185dfab2bSbeveloper *io_count = reply.count; 190285dfab2bSbeveloper if (reply.count > 0) 190385dfab2bSbeveloper memcpy(out_id, reply.node_id, sizeof(media_node_id) * reply.count); 190485dfab2bSbeveloper 190585dfab2bSbeveloper return B_OK; 190685dfab2bSbeveloper } 190752a38012Sejakowatz 190852a38012Sejakowatz 190952a38012Sejakowatz status_t 191052a38012Sejakowatz BMediaRoster::SetRealtimeFlags(uint32 in_enabled) 191152a38012Sejakowatz { 191252a38012Sejakowatz UNIMPLEMENTED(); 191352a38012Sejakowatz return B_ERROR; 191452a38012Sejakowatz } 191552a38012Sejakowatz 191652a38012Sejakowatz 191752a38012Sejakowatz status_t 191852a38012Sejakowatz BMediaRoster::GetRealtimeFlags(uint32 * out_enabled) 191952a38012Sejakowatz { 192052a38012Sejakowatz UNIMPLEMENTED(); 192152a38012Sejakowatz return B_ERROR; 192252a38012Sejakowatz } 192352a38012Sejakowatz 192452a38012Sejakowatz 192552a38012Sejakowatz ssize_t 192652a38012Sejakowatz BMediaRoster::AudioBufferSizeFor(int32 channel_count, 192752a38012Sejakowatz uint32 sample_format, 192852a38012Sejakowatz float frame_rate, 192952a38012Sejakowatz bus_type bus_kind) 193052a38012Sejakowatz { 193152a38012Sejakowatz UNIMPLEMENTED(); 193252a38012Sejakowatz return 4096; 193352a38012Sejakowatz } 193452a38012Sejakowatz 193552a38012Sejakowatz 193652a38012Sejakowatz /* Use MediaFlags to inquire about specific features of the Media Kit. */ 193752a38012Sejakowatz /* Returns < 0 for "not present", positive size for output data size. */ 193852a38012Sejakowatz /* 0 means that the capability is present, but no data about it. */ 193952a38012Sejakowatz /* static */ ssize_t 194052a38012Sejakowatz BMediaRoster::MediaFlags(media_flags cap, 194152a38012Sejakowatz void * buf, 194252a38012Sejakowatz size_t maxSize) 194352a38012Sejakowatz { 194452a38012Sejakowatz UNIMPLEMENTED(); 194552a38012Sejakowatz return 0; 194652a38012Sejakowatz } 194752a38012Sejakowatz 194852a38012Sejakowatz 194952a38012Sejakowatz /* BLooper overrides */ 195052a38012Sejakowatz /* virtual */ void 195152a38012Sejakowatz BMediaRoster::MessageReceived(BMessage * message) 195252a38012Sejakowatz { 19536396865dSbeveloper // media_server plays ping-pong with the BMediaRosters 19546396865dSbeveloper // to detect dead teams. Normal communication uses ports. 19556396865dSbeveloper static BMessage pong('PONG'); 19566396865dSbeveloper if (message->what == 'PING') { 19576396865dSbeveloper message->SendReply(&pong, static_cast<BHandler *>(NULL), 2000000); 19586396865dSbeveloper return; 19596396865dSbeveloper } 19606396865dSbeveloper 19616396865dSbeveloper printf("BMediaRoster::MessageReceived: unknown message!\n"); 19626396865dSbeveloper message->PrintToStream(); 196352a38012Sejakowatz } 196452a38012Sejakowatz 196552a38012Sejakowatz /* virtual */ bool 196652a38012Sejakowatz BMediaRoster::QuitRequested() 196752a38012Sejakowatz { 196852a38012Sejakowatz UNIMPLEMENTED(); 196952a38012Sejakowatz return true; 197052a38012Sejakowatz } 197152a38012Sejakowatz 197252a38012Sejakowatz /* virtual */ BHandler * 197352a38012Sejakowatz BMediaRoster::ResolveSpecifier(BMessage *msg, 197452a38012Sejakowatz int32 index, 197552a38012Sejakowatz BMessage *specifier, 197652a38012Sejakowatz int32 form, 197752a38012Sejakowatz const char *property) 197852a38012Sejakowatz { 197952a38012Sejakowatz UNIMPLEMENTED(); 198052a38012Sejakowatz return 0; 198152a38012Sejakowatz } 198252a38012Sejakowatz 198352a38012Sejakowatz 198452a38012Sejakowatz /* virtual */ status_t 198552a38012Sejakowatz BMediaRoster::GetSupportedSuites(BMessage *data) 198652a38012Sejakowatz { 198752a38012Sejakowatz UNIMPLEMENTED(); 198852a38012Sejakowatz return B_ERROR; 198952a38012Sejakowatz } 199052a38012Sejakowatz 199152a38012Sejakowatz 199252a38012Sejakowatz BMediaRoster::~BMediaRoster() 199352a38012Sejakowatz { 199452a38012Sejakowatz CALLED(); 19956396865dSbeveloper 19966396865dSbeveloper // unregister this application with the media server 19976396865dSbeveloper server_unregister_app_request request; 19986396865dSbeveloper server_unregister_app_reply reply; 19996396865dSbeveloper request.team = team; 20006396865dSbeveloper QueryServer(SERVER_UNREGISTER_APP, &request, sizeof(request), &reply, sizeof(reply)); 200152a38012Sejakowatz } 200252a38012Sejakowatz 200352a38012Sejakowatz 200452a38012Sejakowatz /************************************************************* 200552a38012Sejakowatz * private BMediaRoster 200652a38012Sejakowatz *************************************************************/ 200752a38012Sejakowatz 200852a38012Sejakowatz // deprecated call 200952a38012Sejakowatz status_t 201052a38012Sejakowatz BMediaRoster::SetOutputBuffersFor(const media_source & output, 201152a38012Sejakowatz BBufferGroup * group, 201252a38012Sejakowatz bool will_reclaim ) 201352a38012Sejakowatz { 201452a38012Sejakowatz UNIMPLEMENTED(); 20159e9417d2Sbeveloper debugger("BMediaRoster::SetOutputBuffersFor missing\n"); 201652a38012Sejakowatz return B_ERROR; 201752a38012Sejakowatz } 201852a38012Sejakowatz 201952a38012Sejakowatz 202052a38012Sejakowatz /* FBC stuffing (Mmmh, Stuffing!) */ 202152a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_0(void *) { return B_ERROR; } 202252a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_1(void *) { return B_ERROR; } 202352a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_2(void *) { return B_ERROR; } 202452a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_3(void *) { return B_ERROR; } 202552a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_4(void *) { return B_ERROR; } 202652a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_5(void *) { return B_ERROR; } 202752a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_6(void *) { return B_ERROR; } 202852a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_7(void *) { return B_ERROR; } 202952a38012Sejakowatz 203052a38012Sejakowatz 203152a38012Sejakowatz BMediaRoster::BMediaRoster() : 20326396865dSbeveloper BLooper("_BMediaRoster_", B_URGENT_DISPLAY_PRIORITY, B_LOOPER_PORT_DEFAULT_CAPACITY) 203352a38012Sejakowatz { 203452a38012Sejakowatz CALLED(); 20356396865dSbeveloper 20366396865dSbeveloper // start the looper 20376396865dSbeveloper Run(); 20386396865dSbeveloper 20396396865dSbeveloper // register this application with the media server 20406396865dSbeveloper server_register_app_request request; 20416396865dSbeveloper server_register_app_reply reply; 20426396865dSbeveloper request.team = team; 20436396865dSbeveloper request.messenger = BMessenger(NULL, this); 20446396865dSbeveloper QueryServer(SERVER_REGISTER_APP, &request, sizeof(request), &reply, sizeof(reply)); 204552a38012Sejakowatz } 204652a38012Sejakowatz 20476396865dSbeveloper 204852a38012Sejakowatz /* static */ status_t 204952a38012Sejakowatz BMediaRoster::ParseCommand(BMessage & reply) 205052a38012Sejakowatz { 205152a38012Sejakowatz UNIMPLEMENTED(); 205252a38012Sejakowatz return B_ERROR; 205352a38012Sejakowatz } 205452a38012Sejakowatz 205552a38012Sejakowatz 205652a38012Sejakowatz status_t 205752a38012Sejakowatz BMediaRoster::GetDefaultInfo(media_node_id for_default, 205852a38012Sejakowatz BMessage & out_config) 205952a38012Sejakowatz { 206052a38012Sejakowatz UNIMPLEMENTED(); 206152a38012Sejakowatz return B_ERROR; 206252a38012Sejakowatz } 206352a38012Sejakowatz 206452a38012Sejakowatz 206552a38012Sejakowatz 206652a38012Sejakowatz status_t 206752a38012Sejakowatz BMediaRoster::SetRunningDefault(media_node_id for_default, 206852a38012Sejakowatz const media_node & node) 206952a38012Sejakowatz { 207052a38012Sejakowatz UNIMPLEMENTED(); 207152a38012Sejakowatz return B_ERROR; 207252a38012Sejakowatz } 207352a38012Sejakowatz 207452a38012Sejakowatz 207552a38012Sejakowatz /************************************************************* 207652a38012Sejakowatz * static BMediaRoster variables 207752a38012Sejakowatz *************************************************************/ 207852a38012Sejakowatz 207952a38012Sejakowatz bool BMediaRoster::_isMediaServer; 208052a38012Sejakowatz port_id BMediaRoster::_mReplyPort; 208152a38012Sejakowatz int32 BMediaRoster::_mReplyPortRes; 208252a38012Sejakowatz int32 BMediaRoster::_mReplyPortUnavailCount; 208352a38012Sejakowatz BMediaRoster * BMediaRoster::_sDefault = NULL; 208452a38012Sejakowatz 2085