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 #undef DEBUG 1552a38012Sejakowatz #define DEBUG 3 169e9417d2Sbeveloper #include <Debug.h> 1752a38012Sejakowatz #include "debug.h" 183620737cSbeveloper #include "TStack.h" 1952a38012Sejakowatz #include "PortPool.h" 20*1299bfb2Sbeveloper #include "SystemTimeSource.h" 210a483e72SIngo Weinhold #include "ServerInterface.h" 22f4925104Sbeveloper #include "DataExchange.h" 2316b7eea4Sbeveloper #include "DormantNodeManager.h" 24f4925104Sbeveloper #include "Notifications.h" 2552a38012Sejakowatz 26f4925104Sbeveloper namespace BPrivate { namespace media { 27f4925104Sbeveloper extern team_id team; 283620737cSbeveloper } } // BPrivate::media 293620737cSbeveloper 303620737cSbeveloper using namespace BPrivate::media; 3152a38012Sejakowatz 3252a38012Sejakowatz // the BMediaRoster destructor is private, 3352a38012Sejakowatz // but _DefaultDeleter is a friend class of 3452a38012Sejakowatz // the BMediaRoster an thus can delete it 3552a38012Sejakowatz class _DefaultDeleter 3652a38012Sejakowatz { 3752a38012Sejakowatz public: 38e66fda40Sbeveloper ~_DefaultDeleter() { delete BMediaRoster::_sDefault; } 3952a38012Sejakowatz }; 4052a38012Sejakowatz 41f4925104Sbeveloper _DefaultDeleter _deleter; 4252a38012Sejakowatz 433620737cSbeveloper namespace BPrivate { namespace media { namespace mediaroster { 4452a38012Sejakowatz 4552a38012Sejakowatz status_t GetNode(node_type type, media_node * out_node, int32 * out_input_id = NULL, BString * out_input_name = NULL); 4652a38012Sejakowatz status_t SetNode(node_type type, const media_node *node, const dormant_node_info *info = NULL, const media_input *input = NULL); 473620737cSbeveloper status_t GetAllOutputs(const media_node & node, Stack<media_output> *stack); 483620737cSbeveloper status_t GetAllInputs(const media_node & node, Stack<media_input> *stack); 493620737cSbeveloper status_t PublishOutputs(const media_node & node, Stack<media_output> *stack); 503620737cSbeveloper status_t PublishInputs(const media_node & node, Stack<media_input> *stack); 5152a38012Sejakowatz 523620737cSbeveloper status_t 533620737cSbeveloper GetNode(node_type type, media_node * out_node, int32 * out_input_id, BString * out_input_name) 5452a38012Sejakowatz { 5552a38012Sejakowatz if (out_node == NULL) 5652a38012Sejakowatz return B_BAD_VALUE; 5752a38012Sejakowatz 588c6a6096Sbeveloper server_get_node_request request; 598c6a6096Sbeveloper server_get_node_reply reply; 6052a38012Sejakowatz status_t rv; 6152a38012Sejakowatz 628c6a6096Sbeveloper request.type = type; 639e9417d2Sbeveloper request.team = team; 648c6a6096Sbeveloper rv = QueryServer(SERVER_GET_NODE, &request, sizeof(request), &reply, sizeof(reply)); 658c6a6096Sbeveloper if (rv != B_OK) 668c6a6096Sbeveloper return rv; 6752a38012Sejakowatz 6852a38012Sejakowatz *out_node = reply.node; 6952a38012Sejakowatz if (out_input_id) 7052a38012Sejakowatz *out_input_id = reply.input_id; 7152a38012Sejakowatz if (out_input_name) 7252a38012Sejakowatz *out_input_name = reply.input_name; 738c6a6096Sbeveloper return rv; 7452a38012Sejakowatz } 7552a38012Sejakowatz 763620737cSbeveloper status_t 773620737cSbeveloper SetNode(node_type type, const media_node *node, const dormant_node_info *info, const media_input *input) 7852a38012Sejakowatz { 798c6a6096Sbeveloper server_set_node_request request; 808c6a6096Sbeveloper server_set_node_reply reply; 8152a38012Sejakowatz 828c6a6096Sbeveloper request.type = type; 838c6a6096Sbeveloper request.use_node = node ? true : false; 8452a38012Sejakowatz if (node) 858c6a6096Sbeveloper request.node = *node; 868c6a6096Sbeveloper request.use_dni = info ? true : false; 8752a38012Sejakowatz if (info) 888c6a6096Sbeveloper request.dni = *info; 898c6a6096Sbeveloper request.use_input = input ? true : false; 9052a38012Sejakowatz if (input) 918c6a6096Sbeveloper request.input = *input; 9252a38012Sejakowatz 938c6a6096Sbeveloper return QueryServer(SERVER_SET_NODE, &request, sizeof(request), &reply, sizeof(reply)); 9452a38012Sejakowatz } 9552a38012Sejakowatz 963620737cSbeveloper status_t 973620737cSbeveloper GetAllOutputs(const media_node & node, Stack<media_output> *stack) 983620737cSbeveloper { 993620737cSbeveloper int32 cookie; 1003620737cSbeveloper status_t rv; 1013620737cSbeveloper status_t result; 1023620737cSbeveloper 1033620737cSbeveloper result = B_OK; 1043620737cSbeveloper cookie = 0; 1053620737cSbeveloper for (;;) { 1063620737cSbeveloper producer_get_next_output_request request; 1073620737cSbeveloper producer_get_next_output_reply reply; 1083620737cSbeveloper request.cookie = cookie; 1090caff283Sbeveloper rv = QueryPort(node.port, PRODUCER_GET_NEXT_OUTPUT, &request, sizeof(request), &reply, sizeof(reply)); 1103620737cSbeveloper if (rv != B_OK) 1113620737cSbeveloper break; 1123620737cSbeveloper cookie = reply.cookie; 1133620737cSbeveloper if (!stack->Push(reply.output)) { 1143620737cSbeveloper TRACE("GetAllOutputs: stack->Push failed\n"); 1153620737cSbeveloper result = B_ERROR; 1163620737cSbeveloper } 1173620737cSbeveloper } 1183620737cSbeveloper 1193620737cSbeveloper producer_dispose_output_cookie_request request; 1203620737cSbeveloper producer_dispose_output_cookie_reply reply; 1210caff283Sbeveloper QueryPort(node.port, PRODUCER_DISPOSE_OUTPUT_COOKIE, &request, sizeof(request), &reply, sizeof(reply)); 1223620737cSbeveloper 1233620737cSbeveloper return result; 1243620737cSbeveloper } 1253620737cSbeveloper 1263620737cSbeveloper status_t 1273620737cSbeveloper GetAllInputs(const media_node & node, Stack<media_input> *stack) 1283620737cSbeveloper { 1293620737cSbeveloper int32 cookie; 1303620737cSbeveloper status_t rv; 1313620737cSbeveloper status_t result; 1323620737cSbeveloper 1333620737cSbeveloper result = B_OK; 1343620737cSbeveloper cookie = 0; 1353620737cSbeveloper for (;;) { 1363620737cSbeveloper consumer_get_next_input_request request; 1373620737cSbeveloper consumer_get_next_input_reply reply; 1383620737cSbeveloper request.cookie = cookie; 1390caff283Sbeveloper rv = QueryPort(node.port, CONSUMER_GET_NEXT_INPUT, &request, sizeof(request), &reply, sizeof(reply)); 1403620737cSbeveloper if (rv != B_OK) 1413620737cSbeveloper break; 1423620737cSbeveloper cookie = reply.cookie; 1433620737cSbeveloper if (!stack->Push(reply.input)) { 1443620737cSbeveloper TRACE("GetAllInputs: stack->Push failed\n"); 1453620737cSbeveloper result = B_ERROR; 1463620737cSbeveloper } 1473620737cSbeveloper } 1483620737cSbeveloper 1493620737cSbeveloper consumer_dispose_input_cookie_request request; 1503620737cSbeveloper consumer_dispose_input_cookie_reply reply; 1510caff283Sbeveloper QueryPort(node.port, CONSUMER_DISPOSE_INPUT_COOKIE, &request, sizeof(request), &reply, sizeof(reply)); 1523620737cSbeveloper 1533620737cSbeveloper return result; 1543620737cSbeveloper } 1553620737cSbeveloper 1563620737cSbeveloper status_t 1573620737cSbeveloper PublishOutputs(const media_node & node, Stack<media_output> *stack) 1583620737cSbeveloper { 1593620737cSbeveloper server_publish_outputs_request request; 1603620737cSbeveloper server_publish_outputs_reply reply; 1613620737cSbeveloper media_output *output; 1623620737cSbeveloper media_output *outputs; 1633620737cSbeveloper int32 count; 164a7b41a96Sbeveloper status_t rv; 1653620737cSbeveloper 1663620737cSbeveloper count = stack->CountItems(); 1673620737cSbeveloper TRACE("PublishOutputs: publishing %ld\n", count); 1683620737cSbeveloper 1693620737cSbeveloper request.node = node; 1703620737cSbeveloper request.count = count; 1713620737cSbeveloper if (count > MAX_OUTPUTS) { 1723620737cSbeveloper void *start_addr; 1733620737cSbeveloper size_t size; 1743620737cSbeveloper size = ((count * sizeof(media_output)) + B_PAGE_SIZE - 1) & ~(B_PAGE_SIZE - 1); 1753620737cSbeveloper request.area = create_area("publish outputs", &start_addr, B_ANY_ADDRESS, size, B_NO_LOCK, B_READ_AREA | B_WRITE_AREA); 1763620737cSbeveloper if (request.area < B_OK) { 1773620737cSbeveloper TRACE("PublishOutputs: failed to create area, %#lx\n", request.area); 1783620737cSbeveloper return (status_t)request.area; 1793620737cSbeveloper } 1803620737cSbeveloper outputs = static_cast<media_output *>(start_addr); 1813620737cSbeveloper } else { 1823620737cSbeveloper request.area = -1; 1833620737cSbeveloper outputs = request.outputs; 1843620737cSbeveloper } 1853620737cSbeveloper TRACE("PublishOutputs: area %#lx\n", request.area); 1863620737cSbeveloper 1873620737cSbeveloper for (int32 i = 0; i != count; i++) { 1883620737cSbeveloper stack->GetPointerAt(i, &output); 1893620737cSbeveloper outputs[i] = *output; 1903620737cSbeveloper } 1913620737cSbeveloper 192a7b41a96Sbeveloper rv = QueryServer(SERVER_PUBLISH_OUTPUTS, &request, sizeof(request), &reply, sizeof(reply)); 193a7b41a96Sbeveloper 194a7b41a96Sbeveloper if (request.area != -1) 195a7b41a96Sbeveloper delete_area(request.area); 196a7b41a96Sbeveloper 197a7b41a96Sbeveloper return rv; 1983620737cSbeveloper } 1993620737cSbeveloper 2003620737cSbeveloper status_t 2013620737cSbeveloper PublishInputs(const media_node & node, Stack<media_input> *stack) 2023620737cSbeveloper { 2033620737cSbeveloper server_publish_inputs_request request; 2043620737cSbeveloper server_publish_inputs_reply reply; 2053620737cSbeveloper media_input *input; 2063620737cSbeveloper media_input *inputs; 2073620737cSbeveloper int32 count; 208a7b41a96Sbeveloper status_t rv; 2093620737cSbeveloper 2103620737cSbeveloper count = stack->CountItems(); 2113620737cSbeveloper TRACE("PublishInputs: publishing %ld\n", count); 2123620737cSbeveloper 2133620737cSbeveloper request.node = node; 2143620737cSbeveloper request.count = count; 2153620737cSbeveloper if (count > MAX_INPUTS) { 2163620737cSbeveloper void *start_addr; 2173620737cSbeveloper size_t size; 2183620737cSbeveloper size = ((count * sizeof(media_input)) + B_PAGE_SIZE - 1) & ~(B_PAGE_SIZE - 1); 2193620737cSbeveloper request.area = create_area("publish inputs", &start_addr, B_ANY_ADDRESS, size, B_NO_LOCK, B_READ_AREA | B_WRITE_AREA); 2203620737cSbeveloper if (request.area < B_OK) { 2213620737cSbeveloper TRACE("PublishInputs: failed to create area, %#lx\n", request.area); 2223620737cSbeveloper return (status_t)request.area; 2233620737cSbeveloper } 2243620737cSbeveloper inputs = static_cast<media_input *>(start_addr); 2253620737cSbeveloper } else { 2263620737cSbeveloper request.area = -1; 2273620737cSbeveloper inputs = request.inputs; 2283620737cSbeveloper } 2293620737cSbeveloper TRACE("PublishInputs: area %#lx\n", request.area); 2303620737cSbeveloper 2313620737cSbeveloper for (int32 i = 0; i != count; i++) { 2323620737cSbeveloper stack->GetPointerAt(i, &input); 2333620737cSbeveloper inputs[i] = *input; 2343620737cSbeveloper } 2353620737cSbeveloper 236a7b41a96Sbeveloper rv = QueryServer(SERVER_PUBLISH_INPUTS, &request, sizeof(request), &reply, sizeof(reply)); 237a7b41a96Sbeveloper 238a7b41a96Sbeveloper if (request.area != -1) 239a7b41a96Sbeveloper delete_area(request.area); 240a7b41a96Sbeveloper 241a7b41a96Sbeveloper return rv; 2423620737cSbeveloper } 2433620737cSbeveloper 2443620737cSbeveloper } } } // namespace BPrivate::media::mediaroster 2453620737cSbeveloper 2463620737cSbeveloper using namespace BPrivate::media::mediaroster; 24752a38012Sejakowatz 24852a38012Sejakowatz /************************************************************* 24952a38012Sejakowatz * public BMediaRoster 25052a38012Sejakowatz *************************************************************/ 25152a38012Sejakowatz 25252a38012Sejakowatz status_t 25352a38012Sejakowatz BMediaRoster::GetVideoInput(media_node * out_node) 25452a38012Sejakowatz { 25552a38012Sejakowatz CALLED(); 2563620737cSbeveloper return GetNode(VIDEO_INPUT, out_node); 25752a38012Sejakowatz } 25852a38012Sejakowatz 25952a38012Sejakowatz 26052a38012Sejakowatz status_t 26152a38012Sejakowatz BMediaRoster::GetAudioInput(media_node * out_node) 26252a38012Sejakowatz { 26352a38012Sejakowatz CALLED(); 2643620737cSbeveloper return GetNode(AUDIO_INPUT, out_node); 26552a38012Sejakowatz } 26652a38012Sejakowatz 26752a38012Sejakowatz 26852a38012Sejakowatz status_t 26952a38012Sejakowatz BMediaRoster::GetVideoOutput(media_node * out_node) 27052a38012Sejakowatz { 27152a38012Sejakowatz CALLED(); 2723620737cSbeveloper return GetNode(VIDEO_OUTPUT, out_node); 27352a38012Sejakowatz } 27452a38012Sejakowatz 27552a38012Sejakowatz 27652a38012Sejakowatz status_t 27752a38012Sejakowatz BMediaRoster::GetAudioMixer(media_node * out_node) 27852a38012Sejakowatz { 27952a38012Sejakowatz CALLED(); 2803620737cSbeveloper return GetNode(AUDIO_MIXER, out_node); 28152a38012Sejakowatz } 28252a38012Sejakowatz 28352a38012Sejakowatz 28452a38012Sejakowatz status_t 28552a38012Sejakowatz BMediaRoster::GetAudioOutput(media_node * out_node) 28652a38012Sejakowatz { 28752a38012Sejakowatz CALLED(); 2883620737cSbeveloper return GetNode(AUDIO_OUTPUT, out_node); 28952a38012Sejakowatz } 29052a38012Sejakowatz 29152a38012Sejakowatz 29252a38012Sejakowatz status_t 29352a38012Sejakowatz BMediaRoster::GetAudioOutput(media_node * out_node, 29452a38012Sejakowatz int32 * out_input_id, 29552a38012Sejakowatz BString * out_input_name) 29652a38012Sejakowatz { 29752a38012Sejakowatz CALLED(); 2983620737cSbeveloper return GetNode(AUDIO_OUTPUT_EX, out_node, out_input_id, out_input_name); 29952a38012Sejakowatz } 30052a38012Sejakowatz 30152a38012Sejakowatz 30252a38012Sejakowatz status_t 30352a38012Sejakowatz BMediaRoster::GetTimeSource(media_node * out_node) 30452a38012Sejakowatz { 30552a38012Sejakowatz CALLED(); 3063620737cSbeveloper return GetNode(TIME_SOURCE, out_node); 30752a38012Sejakowatz } 30852a38012Sejakowatz 30952a38012Sejakowatz 31052a38012Sejakowatz status_t 31152a38012Sejakowatz BMediaRoster::SetVideoInput(const media_node & producer) 31252a38012Sejakowatz { 31352a38012Sejakowatz CALLED(); 3143620737cSbeveloper return SetNode(VIDEO_INPUT, &producer); 31552a38012Sejakowatz } 31652a38012Sejakowatz 31752a38012Sejakowatz 31852a38012Sejakowatz status_t 31952a38012Sejakowatz BMediaRoster::SetVideoInput(const dormant_node_info & producer) 32052a38012Sejakowatz { 32152a38012Sejakowatz CALLED(); 3223620737cSbeveloper return SetNode(VIDEO_INPUT, NULL, &producer); 32352a38012Sejakowatz } 32452a38012Sejakowatz 32552a38012Sejakowatz 32652a38012Sejakowatz status_t 32752a38012Sejakowatz BMediaRoster::SetAudioInput(const media_node & producer) 32852a38012Sejakowatz { 32952a38012Sejakowatz CALLED(); 3303620737cSbeveloper return SetNode(AUDIO_INPUT, &producer); 33152a38012Sejakowatz } 33252a38012Sejakowatz 33352a38012Sejakowatz 33452a38012Sejakowatz status_t 33552a38012Sejakowatz BMediaRoster::SetAudioInput(const dormant_node_info & producer) 33652a38012Sejakowatz { 33752a38012Sejakowatz CALLED(); 3383620737cSbeveloper return SetNode(AUDIO_INPUT, NULL, &producer); 33952a38012Sejakowatz } 34052a38012Sejakowatz 34152a38012Sejakowatz 34252a38012Sejakowatz status_t 34352a38012Sejakowatz BMediaRoster::SetVideoOutput(const media_node & consumer) 34452a38012Sejakowatz { 34552a38012Sejakowatz CALLED(); 3463620737cSbeveloper return SetNode(VIDEO_OUTPUT, &consumer); 34752a38012Sejakowatz } 34852a38012Sejakowatz 34952a38012Sejakowatz 35052a38012Sejakowatz status_t 35152a38012Sejakowatz BMediaRoster::SetVideoOutput(const dormant_node_info & consumer) 35252a38012Sejakowatz { 35352a38012Sejakowatz CALLED(); 3543620737cSbeveloper return SetNode(VIDEO_OUTPUT, NULL, &consumer); 35552a38012Sejakowatz } 35652a38012Sejakowatz 35752a38012Sejakowatz 35852a38012Sejakowatz status_t 35952a38012Sejakowatz BMediaRoster::SetAudioOutput(const media_node & consumer) 36052a38012Sejakowatz { 36152a38012Sejakowatz CALLED(); 3623620737cSbeveloper return SetNode(AUDIO_OUTPUT, &consumer); 36352a38012Sejakowatz } 36452a38012Sejakowatz 36552a38012Sejakowatz 36652a38012Sejakowatz status_t 36752a38012Sejakowatz BMediaRoster::SetAudioOutput(const media_input & input_to_output) 36852a38012Sejakowatz { 36952a38012Sejakowatz CALLED(); 3703620737cSbeveloper return SetNode(AUDIO_OUTPUT, NULL, NULL, &input_to_output); 37152a38012Sejakowatz } 37252a38012Sejakowatz 37352a38012Sejakowatz 37452a38012Sejakowatz status_t 37552a38012Sejakowatz BMediaRoster::SetAudioOutput(const dormant_node_info & consumer) 37652a38012Sejakowatz { 37752a38012Sejakowatz CALLED(); 3783620737cSbeveloper return SetNode(AUDIO_OUTPUT, NULL, &consumer); 37952a38012Sejakowatz } 38052a38012Sejakowatz 38152a38012Sejakowatz 38252a38012Sejakowatz status_t 38352a38012Sejakowatz BMediaRoster::GetNodeFor(media_node_id node, 38452a38012Sejakowatz media_node * clone) 38552a38012Sejakowatz { 3869e9417d2Sbeveloper CALLED(); 3879e9417d2Sbeveloper if (clone == NULL) 3889e9417d2Sbeveloper return B_BAD_VALUE; 3899e9417d2Sbeveloper if (node <= 0) 3909e9417d2Sbeveloper return B_MEDIA_BAD_NODE; 3919e9417d2Sbeveloper 3929e9417d2Sbeveloper server_get_node_for_request request; 3939e9417d2Sbeveloper server_get_node_for_reply reply; 3949e9417d2Sbeveloper status_t rv; 3959e9417d2Sbeveloper 3969e9417d2Sbeveloper request.nodeid = node; 3979e9417d2Sbeveloper request.team = team; 3989e9417d2Sbeveloper 3999e9417d2Sbeveloper rv = QueryServer(SERVER_GET_NODE_FOR, &request, sizeof(request), &reply, sizeof(reply)); 4009e9417d2Sbeveloper if (rv != B_OK) 4019e9417d2Sbeveloper return rv; 4029e9417d2Sbeveloper 4039e9417d2Sbeveloper *clone = reply.clone; 4049e9417d2Sbeveloper return B_OK; 40552a38012Sejakowatz } 40652a38012Sejakowatz 40752a38012Sejakowatz 40852a38012Sejakowatz status_t 40952a38012Sejakowatz BMediaRoster::GetSystemTimeSource(media_node * clone) 41052a38012Sejakowatz { 41152a38012Sejakowatz CALLED(); 4123620737cSbeveloper return GetNode(SYSTEM_TIME_SOURCE, clone); 41352a38012Sejakowatz } 41452a38012Sejakowatz 41552a38012Sejakowatz 41652a38012Sejakowatz status_t 41752a38012Sejakowatz BMediaRoster::ReleaseNode(const media_node & node) 41852a38012Sejakowatz { 4199e9417d2Sbeveloper CALLED(); 4209e9417d2Sbeveloper if (node.node <= 0) 4219e9417d2Sbeveloper return B_MEDIA_BAD_NODE; 42252a38012Sejakowatz 4239e9417d2Sbeveloper server_release_node_request request; 4249e9417d2Sbeveloper server_release_node_reply reply; 4259e9417d2Sbeveloper 4269e9417d2Sbeveloper request.node = node; 4279e9417d2Sbeveloper request.team = team; 4289e9417d2Sbeveloper 4299e9417d2Sbeveloper return QueryServer(SERVER_RELEASE_NODE, &request, sizeof(request), &reply, sizeof(reply)); 4309e9417d2Sbeveloper } 43152a38012Sejakowatz 43252a38012Sejakowatz 43352a38012Sejakowatz BTimeSource * 43452a38012Sejakowatz BMediaRoster::MakeTimeSourceFor(const media_node & for_node) 43552a38012Sejakowatz { 436*1299bfb2Sbeveloper BROKEN(); 437*1299bfb2Sbeveloper return new _SysTimeSource(); // XXX fix this 43852a38012Sejakowatz } 43952a38012Sejakowatz 44052a38012Sejakowatz 44152a38012Sejakowatz status_t 44252a38012Sejakowatz BMediaRoster::Connect(const media_source & from, 44352a38012Sejakowatz const media_destination & to, 44452a38012Sejakowatz media_format * io_format, 44552a38012Sejakowatz media_output * out_output, 44652a38012Sejakowatz media_input * out_input) 44752a38012Sejakowatz { 448eae26d3dSbeveloper return BMediaRoster::Connect(from, to, io_format, out_output, out_input, 0); 449eae26d3dSbeveloper } 450eae26d3dSbeveloper 451eae26d3dSbeveloper 452eae26d3dSbeveloper status_t 453eae26d3dSbeveloper BMediaRoster::Connect(const media_source & from, 454eae26d3dSbeveloper const media_destination & to, 455eae26d3dSbeveloper media_format * io_format, 456eae26d3dSbeveloper media_output * out_output, 457eae26d3dSbeveloper media_input * out_input, 458eae26d3dSbeveloper uint32 in_flags, 459eae26d3dSbeveloper void * _reserved) 460eae26d3dSbeveloper { 46152a38012Sejakowatz CALLED(); 46252a38012Sejakowatz if (io_format == NULL || out_output == NULL || out_input == NULL) 46352a38012Sejakowatz return B_BAD_VALUE; 4649e9417d2Sbeveloper if (from == media_source::null) { 4659e9417d2Sbeveloper TRACE("BMediaRoster::Connect: media_source invalid\n"); 46652a38012Sejakowatz return B_MEDIA_BAD_SOURCE; 4679e9417d2Sbeveloper } 4689e9417d2Sbeveloper if (to == media_destination::null) { 4699e9417d2Sbeveloper TRACE("BMediaRoster::Connect: media_destination invalid\n"); 47052a38012Sejakowatz return B_MEDIA_BAD_DESTINATION; 4719e9417d2Sbeveloper } 47252a38012Sejakowatz 47352a38012Sejakowatz status_t rv; 4748c6a6096Sbeveloper producer_format_proposal_request request1; 4758c6a6096Sbeveloper producer_format_proposal_reply reply1; 47652a38012Sejakowatz 47752a38012Sejakowatz // BBufferProducer::FormatProposal 4788c6a6096Sbeveloper request1.output = from; 4798c6a6096Sbeveloper request1.format = *io_format; 4808c6a6096Sbeveloper rv = QueryPort(from.port, PRODUCER_FORMAT_PROPOSAL, &request1, sizeof(request1), &reply1, sizeof(reply1)); 4818c6a6096Sbeveloper if (rv != B_OK) { 4825edd2016Sbeveloper TRACE("BMediaRoster::Connect: aborted after BBufferProducer::FormatProposal, status = %#lx\n",rv); 48352a38012Sejakowatz return rv; 48452a38012Sejakowatz } 4858c6a6096Sbeveloper // reply1.format now contains the format proposed by the producer 4868c6a6096Sbeveloper 4878c6a6096Sbeveloper consumer_accept_format_request request2; 4888c6a6096Sbeveloper consumer_accept_format_reply reply2; 4898c6a6096Sbeveloper 4908c6a6096Sbeveloper // BBufferConsumer::AcceptFormat 4918c6a6096Sbeveloper request2.dest = to; 4928c6a6096Sbeveloper request2.format = reply1.format; 4938c6a6096Sbeveloper rv = QueryPort(to.port, CONSUMER_ACCEPT_FORMAT, &request2, sizeof(request2), &reply2, sizeof(reply2)); 4948c6a6096Sbeveloper if (rv != B_OK) { 4955edd2016Sbeveloper TRACE("BMediaRoster::Connect: aborted after BBufferConsumer::AcceptFormat, status = %#lx\n",rv); 4968c6a6096Sbeveloper return rv; 4978c6a6096Sbeveloper } 4988c6a6096Sbeveloper // reply2.format now contains the format accepted by the consumer 4998c6a6096Sbeveloper 5008c6a6096Sbeveloper // BBufferProducer::PrepareToConnect 5018c6a6096Sbeveloper producer_prepare_to_connect_request request3; 5028c6a6096Sbeveloper producer_prepare_to_connect_reply reply3; 5038c6a6096Sbeveloper 5048c6a6096Sbeveloper request3.source = from; 5058c6a6096Sbeveloper request3.destination = to; 5068c6a6096Sbeveloper request3.format = reply2.format; 5078c6a6096Sbeveloper strcpy(request3.name, "XXX some default name"); // XXX fix this 5088c6a6096Sbeveloper rv = QueryPort(from.port, PRODUCER_PREPARE_TO_CONNECT, &request3, sizeof(request3), &reply3, sizeof(reply3)); 5098c6a6096Sbeveloper if (rv != B_OK) { 5105edd2016Sbeveloper TRACE("BMediaRoster::Connect: aborted after BBufferProducer::PrepareToConnect, status = %#lx\n",rv); 5118c6a6096Sbeveloper return rv; 5128c6a6096Sbeveloper } 5138c6a6096Sbeveloper // reply3.format is still our pretty media format 5148c6a6096Sbeveloper // reply3.out_source the real source to be used for the connection 5158c6a6096Sbeveloper // reply3.name the name BBufferConsumer::Connected will see in the outInput->name argument 5168c6a6096Sbeveloper 5178c6a6096Sbeveloper // BBufferConsumer::Connected 5188c6a6096Sbeveloper consumer_connected_request request4; 5198c6a6096Sbeveloper consumer_connected_reply reply4; 5208c6a6096Sbeveloper status_t con_status; 5218c6a6096Sbeveloper 5228c6a6096Sbeveloper request4.producer = reply3.out_source; 5238c6a6096Sbeveloper request4.where = to; 5248c6a6096Sbeveloper request4.with_format = reply3.format; 5258c6a6096Sbeveloper con_status = QueryPort(to.port, CONSUMER_CONNECTED, &request4, sizeof(request4), &reply4, sizeof(reply4)); 5268c6a6096Sbeveloper if (con_status != B_OK) { 5275edd2016Sbeveloper TRACE("BMediaRoster::Connect: aborting after BBufferConsumer::Connected, status = %#lx\n",con_status); 5288c6a6096Sbeveloper // we do NOT return here! 5298c6a6096Sbeveloper } 5308c6a6096Sbeveloper // con_status contains the status code to be supplied to BBufferProducer::Connect's status argument 5318c6a6096Sbeveloper // reply4.input contains the media_input that describes the connection from the consumer point of view 5328c6a6096Sbeveloper 5338c6a6096Sbeveloper // BBufferProducer::Connect 5348c6a6096Sbeveloper producer_connect_request request5; 5358c6a6096Sbeveloper producer_connect_reply reply5; 5368c6a6096Sbeveloper 5378c6a6096Sbeveloper request5.error = con_status; 5388c6a6096Sbeveloper request5.source = reply3.out_source; 5398c6a6096Sbeveloper request5.destination = reply4.input.destination; 5408c6a6096Sbeveloper request5.format = reply3.format; // XXX reply4.input.format ??? 5418c6a6096Sbeveloper strcpy(request5.name, reply4.input.name); 5420caff283Sbeveloper rv = QueryPort(reply4.input.source.port, PRODUCER_CONNECT, &request5, sizeof(request5), &reply5, sizeof(reply5)); 5438c6a6096Sbeveloper if (con_status != B_OK) { 5448c6a6096Sbeveloper TRACE("BMediaRoster::Connect: aborted\n"); 5458c6a6096Sbeveloper return con_status; 5468c6a6096Sbeveloper } 5478c6a6096Sbeveloper if (rv != B_OK) { 5485edd2016Sbeveloper TRACE("BMediaRoster::Connect: aborted after BBufferProducer::Connect, status = %#lx\n",rv); 5498c6a6096Sbeveloper return rv; 5508c6a6096Sbeveloper } 5518c6a6096Sbeveloper // reply5.name contains the name assigned to the connection by the producer 5528c6a6096Sbeveloper 5538c6a6096Sbeveloper // find the output node 5548c6a6096Sbeveloper // XXX isn't there a easier way? 5558c6a6096Sbeveloper media_node sourcenode; 5568c6a6096Sbeveloper GetNodeFor(NodeIDFor(from.port), &sourcenode); 5578c6a6096Sbeveloper ReleaseNode(sourcenode); 5588c6a6096Sbeveloper 5598c6a6096Sbeveloper // initilize connection info 5608c6a6096Sbeveloper *io_format = reply3.format; 5618c6a6096Sbeveloper *out_input = reply4.input; 5628c6a6096Sbeveloper out_output->node = sourcenode; 5638c6a6096Sbeveloper out_output->source = reply4.input.source; 5648c6a6096Sbeveloper out_output->destination = reply4.input.destination; 5658c6a6096Sbeveloper out_output->format = reply4.input.format; 5668c6a6096Sbeveloper strcpy(out_output->name, reply5.name); 5678c6a6096Sbeveloper 5688c6a6096Sbeveloper // the connection is now made 5698c6a6096Sbeveloper 5708c6a6096Sbeveloper 5718c6a6096Sbeveloper // XXX register connection with server 5729e9417d2Sbeveloper // XXX we should just send a notification, instead of republishing all endpoints 5739e9417d2Sbeveloper Stack<media_output> outstack; 5749e9417d2Sbeveloper Stack<media_input> instack; 5759e9417d2Sbeveloper if (B_OK == GetAllOutputs(out_output->node , &outstack)) 5769e9417d2Sbeveloper PublishOutputs(out_output->node , &outstack); 5779e9417d2Sbeveloper if (B_OK == GetAllInputs(out_input->node , &instack)) 5789e9417d2Sbeveloper PublishInputs(out_input->node, &instack); 5798c6a6096Sbeveloper 5808c6a6096Sbeveloper 5818c6a6096Sbeveloper // XXX if (mute) BBufferProducer::EnableOutput(false) 5829e9417d2Sbeveloper if (in_flags & B_CONNECT_MUTED) { 5839e9417d2Sbeveloper } 5848c6a6096Sbeveloper 5858c6a6096Sbeveloper 5869e9417d2Sbeveloper // send a notification 5879e9417d2Sbeveloper BPrivate::media::notifications::ConnectionMade(*out_input, *out_output, *io_format); 5889e9417d2Sbeveloper 5898c6a6096Sbeveloper return B_OK; 5908c6a6096Sbeveloper }; 59152a38012Sejakowatz 59252a38012Sejakowatz 59352a38012Sejakowatz status_t 5949e9417d2Sbeveloper BMediaRoster::Disconnect(media_node_id source_nodeid, 59552a38012Sejakowatz const media_source & source, 5969e9417d2Sbeveloper media_node_id destination_nodeid, 59752a38012Sejakowatz const media_destination & destination) 59852a38012Sejakowatz { 5999e9417d2Sbeveloper CALLED(); 6009e9417d2Sbeveloper if (source_nodeid <= 0) { 6019e9417d2Sbeveloper TRACE("BMediaRoster::Disconnect: source media_node_id invalid\n"); 6029e9417d2Sbeveloper return B_MEDIA_BAD_SOURCE; 6039e9417d2Sbeveloper } 6049e9417d2Sbeveloper if (destination_nodeid <= 0) { 6059e9417d2Sbeveloper TRACE("BMediaRoster::Disconnect: source media_node_id invalid\n"); 6069e9417d2Sbeveloper return B_MEDIA_BAD_DESTINATION; 6079e9417d2Sbeveloper } 6089e9417d2Sbeveloper if (source == media_source::null) { 6099e9417d2Sbeveloper TRACE("BMediaRoster::Disconnect: media_source invalid\n"); 6109e9417d2Sbeveloper return B_MEDIA_BAD_SOURCE; 6119e9417d2Sbeveloper } 6129e9417d2Sbeveloper if (destination == media_destination::null) { 6139e9417d2Sbeveloper TRACE("BMediaRoster::Disconnect: media_destination invalid\n"); 6149e9417d2Sbeveloper return B_MEDIA_BAD_DESTINATION; 6159e9417d2Sbeveloper } 6169e9417d2Sbeveloper 6179e9417d2Sbeveloper producer_disconnect_request request2; 6189e9417d2Sbeveloper producer_disconnect_reply reply2; 6199e9417d2Sbeveloper consumer_disconnected_request request1; 6209e9417d2Sbeveloper consumer_disconnected_reply reply1; 6219e9417d2Sbeveloper status_t rv1, rv2; 6229e9417d2Sbeveloper 6239e9417d2Sbeveloper // XXX we should ask the server if this connection really exists 6249e9417d2Sbeveloper 6259e9417d2Sbeveloper request1.source = source; 6269e9417d2Sbeveloper request1.destination = destination; 6279e9417d2Sbeveloper request2.source = source; 6289e9417d2Sbeveloper request2.destination = destination; 6299e9417d2Sbeveloper 6309e9417d2Sbeveloper rv1 = QueryPort(source.port, PRODUCER_DISCONNECT, &request1, sizeof(request1), &reply1, sizeof(reply1)); 6319e9417d2Sbeveloper rv2 = QueryPort(destination.port, CONSUMER_DISCONNECTED, &request2, sizeof(request2), &reply2, sizeof(reply2)); 6329e9417d2Sbeveloper 6339e9417d2Sbeveloper // XXX unregister connection with server 6349e9417d2Sbeveloper // XXX we should just send a notification, instead of republishing all endpoints 6359e9417d2Sbeveloper Stack<media_output> outstack; 6369e9417d2Sbeveloper Stack<media_input> instack; 6379e9417d2Sbeveloper media_node sourcenode; 6389e9417d2Sbeveloper media_node destnode; 6399e9417d2Sbeveloper if (B_OK == GetNodeFor(source_nodeid, &sourcenode)) { 6409e9417d2Sbeveloper if (B_OK == GetAllOutputs(sourcenode , &outstack)) 6419e9417d2Sbeveloper PublishOutputs(sourcenode , &outstack); 6429e9417d2Sbeveloper ReleaseNode(sourcenode); 6439e9417d2Sbeveloper } else TRACE("BMediaRoster::Disconnect: source GetNodeFor failed\n"); 6449e9417d2Sbeveloper if (B_OK == GetNodeFor(destination_nodeid, &destnode)) { 6459e9417d2Sbeveloper if (B_OK == GetAllInputs(destnode , &instack)) 6469e9417d2Sbeveloper PublishInputs(destnode, &instack); 6479e9417d2Sbeveloper ReleaseNode(destnode); 6489e9417d2Sbeveloper } else TRACE("BMediaRoster::Disconnect: dest GetNodeFor failed\n"); 6499e9417d2Sbeveloper 6509e9417d2Sbeveloper 6519e9417d2Sbeveloper // send a notification 6529e9417d2Sbeveloper BPrivate::media::notifications::ConnectionBroken(source, destination); 6539e9417d2Sbeveloper 6549e9417d2Sbeveloper return (rv1 != B_OK || rv2 != B_OK) ? B_ERROR : B_OK; 65552a38012Sejakowatz } 65652a38012Sejakowatz 65752a38012Sejakowatz 65852a38012Sejakowatz status_t 65952a38012Sejakowatz BMediaRoster::StartNode(const media_node & node, 66052a38012Sejakowatz bigtime_t at_performance_time) 66152a38012Sejakowatz { 66252a38012Sejakowatz CALLED(); 6639e9417d2Sbeveloper if (node.node <= 0) 66452a38012Sejakowatz return B_MEDIA_BAD_NODE; 66552a38012Sejakowatz 66652a38012Sejakowatz xfer_node_start msg; 66752a38012Sejakowatz msg.performance_time = at_performance_time; 66852a38012Sejakowatz 66952a38012Sejakowatz return write_port(node.port, NODE_START, &msg, sizeof(msg)); 67052a38012Sejakowatz } 67152a38012Sejakowatz 67252a38012Sejakowatz 67352a38012Sejakowatz status_t 67452a38012Sejakowatz BMediaRoster::StopNode(const media_node & node, 67552a38012Sejakowatz bigtime_t at_performance_time, 67652a38012Sejakowatz bool immediate) 67752a38012Sejakowatz { 67852a38012Sejakowatz CALLED(); 6799e9417d2Sbeveloper if (node.node <= 0) 68052a38012Sejakowatz return B_MEDIA_BAD_NODE; 68152a38012Sejakowatz 68252a38012Sejakowatz xfer_node_stop msg; 68352a38012Sejakowatz msg.performance_time = at_performance_time; 68452a38012Sejakowatz msg.immediate = immediate; 68552a38012Sejakowatz 68652a38012Sejakowatz return write_port(node.port, NODE_STOP, &msg, sizeof(msg)); 68752a38012Sejakowatz } 68852a38012Sejakowatz 68952a38012Sejakowatz 69052a38012Sejakowatz status_t 69152a38012Sejakowatz BMediaRoster::SeekNode(const media_node & node, 69252a38012Sejakowatz bigtime_t to_media_time, 69352a38012Sejakowatz bigtime_t at_performance_time) 69452a38012Sejakowatz { 69552a38012Sejakowatz CALLED(); 6969e9417d2Sbeveloper if (node.node <= 0) 69752a38012Sejakowatz return B_MEDIA_BAD_NODE; 69852a38012Sejakowatz 69952a38012Sejakowatz xfer_node_seek msg; 70052a38012Sejakowatz msg.media_time = to_media_time; 70152a38012Sejakowatz msg.performance_time = at_performance_time; 70252a38012Sejakowatz 70352a38012Sejakowatz return write_port(node.port, NODE_SEEK, &msg, sizeof(msg)); 70452a38012Sejakowatz } 70552a38012Sejakowatz 70652a38012Sejakowatz 70752a38012Sejakowatz status_t 70852a38012Sejakowatz BMediaRoster::StartTimeSource(const media_node & node, 70952a38012Sejakowatz bigtime_t at_real_time) 71052a38012Sejakowatz { 71152a38012Sejakowatz CALLED(); 7129e9417d2Sbeveloper if (node.node <= 0) 71352a38012Sejakowatz return B_MEDIA_BAD_NODE; 71452a38012Sejakowatz if ((node.kind & B_TIME_SOURCE) == 0) 71552a38012Sejakowatz return B_MEDIA_BAD_NODE; 71652a38012Sejakowatz 71752a38012Sejakowatz BTimeSource::time_source_op_info msg; 71852a38012Sejakowatz msg.op = BTimeSource::B_TIMESOURCE_START; 71952a38012Sejakowatz msg.real_time = at_real_time; 72052a38012Sejakowatz 72152a38012Sejakowatz return write_port(node.port, TIMESOURCE_OP, &msg, sizeof(msg)); 72252a38012Sejakowatz } 72352a38012Sejakowatz 72452a38012Sejakowatz 72552a38012Sejakowatz status_t 72652a38012Sejakowatz BMediaRoster::StopTimeSource(const media_node & node, 72752a38012Sejakowatz bigtime_t at_real_time, 72852a38012Sejakowatz bool immediate) 72952a38012Sejakowatz { 73052a38012Sejakowatz CALLED(); 7319e9417d2Sbeveloper if (node.node <= 0) 73252a38012Sejakowatz return B_MEDIA_BAD_NODE; 73352a38012Sejakowatz if ((node.kind & B_TIME_SOURCE) == 0) 73452a38012Sejakowatz return B_MEDIA_BAD_NODE; 73552a38012Sejakowatz 73652a38012Sejakowatz BTimeSource::time_source_op_info msg; 73752a38012Sejakowatz msg.op = immediate ? BTimeSource::B_TIMESOURCE_STOP_IMMEDIATELY : BTimeSource::B_TIMESOURCE_STOP; 73852a38012Sejakowatz msg.real_time = at_real_time; 73952a38012Sejakowatz 74052a38012Sejakowatz return write_port(node.port, TIMESOURCE_OP, &msg, sizeof(msg)); 74152a38012Sejakowatz } 74252a38012Sejakowatz 74352a38012Sejakowatz 74452a38012Sejakowatz status_t 74552a38012Sejakowatz BMediaRoster::SeekTimeSource(const media_node & node, 74652a38012Sejakowatz bigtime_t to_performance_time, 74752a38012Sejakowatz bigtime_t at_real_time) 74852a38012Sejakowatz { 74952a38012Sejakowatz CALLED(); 7509e9417d2Sbeveloper if (node.node <= 0) 75152a38012Sejakowatz return B_MEDIA_BAD_NODE; 75252a38012Sejakowatz if ((node.kind & B_TIME_SOURCE) == 0) 75352a38012Sejakowatz return B_MEDIA_BAD_NODE; 75452a38012Sejakowatz 75552a38012Sejakowatz BTimeSource::time_source_op_info msg; 75652a38012Sejakowatz msg.op = BTimeSource::B_TIMESOURCE_SEEK; 75752a38012Sejakowatz msg.real_time = at_real_time; 75852a38012Sejakowatz msg.performance_time = to_performance_time; 75952a38012Sejakowatz 76052a38012Sejakowatz return write_port(node.port, TIMESOURCE_OP, &msg, sizeof(msg)); 76152a38012Sejakowatz } 76252a38012Sejakowatz 76352a38012Sejakowatz 76452a38012Sejakowatz status_t 76552a38012Sejakowatz BMediaRoster::SyncToNode(const media_node & node, 76652a38012Sejakowatz bigtime_t at_time, 76752a38012Sejakowatz bigtime_t timeout) 76852a38012Sejakowatz { 76952a38012Sejakowatz UNIMPLEMENTED(); 77052a38012Sejakowatz return B_ERROR; 77152a38012Sejakowatz } 77252a38012Sejakowatz 77352a38012Sejakowatz 77452a38012Sejakowatz status_t 77552a38012Sejakowatz BMediaRoster::SetRunModeNode(const media_node & node, 77652a38012Sejakowatz BMediaNode::run_mode mode) 77752a38012Sejakowatz { 77852a38012Sejakowatz CALLED(); 7799e9417d2Sbeveloper if (node.node <= 0) 78052a38012Sejakowatz return B_MEDIA_BAD_NODE; 78152a38012Sejakowatz 78252a38012Sejakowatz xfer_node_set_run_mode msg; 78352a38012Sejakowatz msg.mode = mode; 78452a38012Sejakowatz 78552a38012Sejakowatz return write_port(node.port, NODE_SET_RUN_MODE, &msg, sizeof(msg)); 78652a38012Sejakowatz } 78752a38012Sejakowatz 78852a38012Sejakowatz 78952a38012Sejakowatz status_t 79052a38012Sejakowatz BMediaRoster::PrerollNode(const media_node & node) 79152a38012Sejakowatz { 79252a38012Sejakowatz CALLED(); 7939e9417d2Sbeveloper if (node.node <= 0) 79452a38012Sejakowatz return B_MEDIA_BAD_NODE; 79552a38012Sejakowatz 79652a38012Sejakowatz char dummy; 79752a38012Sejakowatz return write_port(node.port, NODE_PREROLL, &dummy, sizeof(dummy)); 79852a38012Sejakowatz } 79952a38012Sejakowatz 80052a38012Sejakowatz 80152a38012Sejakowatz status_t 80252a38012Sejakowatz BMediaRoster::RollNode(const media_node & node, 80352a38012Sejakowatz bigtime_t startPerformance, 80452a38012Sejakowatz bigtime_t stopPerformance, 80552a38012Sejakowatz bigtime_t atMediaTime) 80652a38012Sejakowatz { 80752a38012Sejakowatz UNIMPLEMENTED(); 80852a38012Sejakowatz return B_ERROR; 80952a38012Sejakowatz } 81052a38012Sejakowatz 81152a38012Sejakowatz 81252a38012Sejakowatz status_t 81352a38012Sejakowatz BMediaRoster::SetProducerRunModeDelay(const media_node & node, 81452a38012Sejakowatz bigtime_t delay, 81552a38012Sejakowatz BMediaNode::run_mode mode) 81652a38012Sejakowatz { 81752a38012Sejakowatz UNIMPLEMENTED(); 81852a38012Sejakowatz return B_ERROR; 81952a38012Sejakowatz } 82052a38012Sejakowatz 82152a38012Sejakowatz 82252a38012Sejakowatz status_t 82352a38012Sejakowatz BMediaRoster::SetProducerRate(const media_node & producer, 82452a38012Sejakowatz int32 numer, 82552a38012Sejakowatz int32 denom) 82652a38012Sejakowatz { 82752a38012Sejakowatz CALLED(); 82852a38012Sejakowatz if (producer.node == 0) 82952a38012Sejakowatz return B_MEDIA_BAD_NODE; 83052a38012Sejakowatz if ((producer.kind & B_BUFFER_PRODUCER) == 0) 83152a38012Sejakowatz return B_MEDIA_BAD_NODE; 83252a38012Sejakowatz 83352a38012Sejakowatz xfer_producer_set_play_rate msg; 83452a38012Sejakowatz xfer_producer_set_play_rate_reply reply; 83552a38012Sejakowatz status_t rv; 83652a38012Sejakowatz int32 code; 83752a38012Sejakowatz 83852a38012Sejakowatz msg.numer = numer; 83952a38012Sejakowatz msg.denom = denom; 84052a38012Sejakowatz msg.reply_port = _PortPool->GetPort(); 84152a38012Sejakowatz rv = write_port(producer.node, PRODUCER_SET_PLAY_RATE, &msg, sizeof(msg)); 84252a38012Sejakowatz if (rv != B_OK) { 84352a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 84452a38012Sejakowatz return rv; 84552a38012Sejakowatz } 84652a38012Sejakowatz rv = read_port(msg.reply_port, &code, &reply, sizeof(reply)); 84752a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 84852a38012Sejakowatz return (rv < B_OK) ? rv : reply.result; 84952a38012Sejakowatz } 85052a38012Sejakowatz 85152a38012Sejakowatz 85252a38012Sejakowatz /* Nodes will have available inputs/outputs as long as they are capable */ 85352a38012Sejakowatz /* of accepting more connections. The node may create an additional */ 85452a38012Sejakowatz /* output or input as the currently available is taken into usage. */ 85552a38012Sejakowatz status_t 85652a38012Sejakowatz BMediaRoster::GetLiveNodeInfo(const media_node & node, 85752a38012Sejakowatz live_node_info * out_live_info) 85852a38012Sejakowatz { 8599e9417d2Sbeveloper CALLED(); 8609e9417d2Sbeveloper if (out_live_info == NULL) 8619e9417d2Sbeveloper return B_BAD_VALUE; 8629e9417d2Sbeveloper if (node.node <= 0) 8639e9417d2Sbeveloper return B_MEDIA_BAD_NODE; 8649e9417d2Sbeveloper 8659e9417d2Sbeveloper server_get_live_node_info_request request; 8669e9417d2Sbeveloper server_get_live_node_info_reply reply; 8679e9417d2Sbeveloper status_t rv; 8689e9417d2Sbeveloper 8699e9417d2Sbeveloper request.node = node; 8709e9417d2Sbeveloper 871*1299bfb2Sbeveloper rv = QueryServer(SERVER_GET_LIVE_NODE_INFO, &request, sizeof(request), &reply, sizeof(reply)); 8729e9417d2Sbeveloper if (rv != B_OK) 8739e9417d2Sbeveloper return rv; 8749e9417d2Sbeveloper 8759e9417d2Sbeveloper *out_live_info = reply.live_info; 8769e9417d2Sbeveloper return B_OK; 87752a38012Sejakowatz } 87852a38012Sejakowatz 87952a38012Sejakowatz 88052a38012Sejakowatz status_t 88152a38012Sejakowatz BMediaRoster::GetLiveNodes(live_node_info * out_live_nodes, 88252a38012Sejakowatz int32 * io_total_count, 88352a38012Sejakowatz const media_format * has_input, 88452a38012Sejakowatz const media_format * has_output, 88552a38012Sejakowatz const char * name, 88652a38012Sejakowatz uint64 node_kinds) 88752a38012Sejakowatz { 8889e9417d2Sbeveloper CALLED(); 8899e9417d2Sbeveloper if (out_live_nodes == NULL || io_total_count == NULL) 8909e9417d2Sbeveloper return B_BAD_VALUE; 8919e9417d2Sbeveloper if (*io_total_count <= 0) 8929e9417d2Sbeveloper return B_BAD_VALUE; 8939e9417d2Sbeveloper 8949e9417d2Sbeveloper // XXX we also support the wildcard search as GetDormantNodes does. This needs to be documented 8959e9417d2Sbeveloper 8969e9417d2Sbeveloper server_get_live_nodes_request request; 8979e9417d2Sbeveloper server_get_live_nodes_reply reply; 8989e9417d2Sbeveloper status_t rv; 8999e9417d2Sbeveloper 9009e9417d2Sbeveloper request.maxcount = *io_total_count; 9019e9417d2Sbeveloper request.has_input = (bool) has_input; 9029e9417d2Sbeveloper if (has_input) 9039e9417d2Sbeveloper request.inputformat = *has_input; // XXX we should not make a flat copy of media_format 9049e9417d2Sbeveloper request.has_output = (bool) has_output; 9059e9417d2Sbeveloper if (has_output) 9069e9417d2Sbeveloper request.outputformat = *has_output; // XXX we should not make a flat copy of media_format 9079e9417d2Sbeveloper request.has_name = (bool) name; 9089e9417d2Sbeveloper if (name) { 9099e9417d2Sbeveloper int len = strlen(name); 9109e9417d2Sbeveloper len = min_c(len, (int)sizeof(request.name) - 1); 9119e9417d2Sbeveloper memcpy(request.name, name, len); 9129e9417d2Sbeveloper request.name[len] = 0; 9139e9417d2Sbeveloper } 9149e9417d2Sbeveloper request.require_kinds = node_kinds; 9159e9417d2Sbeveloper 9169e9417d2Sbeveloper rv = QueryServer(SERVER_GET_LIVE_NODES, &request, sizeof(request), &reply, sizeof(reply)); 9179e9417d2Sbeveloper if (rv != B_OK) { 9189e9417d2Sbeveloper TRACE("BMediaRoster::GetLiveNodes failed\n"); 919*1299bfb2Sbeveloper *io_total_count = 0; 9209e9417d2Sbeveloper return rv; 9219e9417d2Sbeveloper } 9229e9417d2Sbeveloper 9239e9417d2Sbeveloper if (reply.count > MAX_LIVE_INFO) { 9249e9417d2Sbeveloper live_node_info *live_info; 9259e9417d2Sbeveloper area_id clone; 9269e9417d2Sbeveloper 9279e9417d2Sbeveloper clone = clone_area("live_node_info clone", reinterpret_cast<void **>(&live_info), B_ANY_ADDRESS, B_READ_AREA | B_WRITE_AREA, reply.area); 9289e9417d2Sbeveloper if (clone < B_OK) { 9299e9417d2Sbeveloper TRACE("BMediaRoster::GetLiveNodes failed to clone area, %#lx\n", clone); 9309e9417d2Sbeveloper delete_area(reply.area); 931*1299bfb2Sbeveloper *io_total_count = 0; 93252a38012Sejakowatz return B_ERROR; 93352a38012Sejakowatz } 93452a38012Sejakowatz 9359e9417d2Sbeveloper for (int32 i = 0; i < reply.count; i++) { 9369e9417d2Sbeveloper out_live_nodes[i] = live_info[i]; 9379e9417d2Sbeveloper } 9389e9417d2Sbeveloper 9399e9417d2Sbeveloper delete_area(clone); 9409e9417d2Sbeveloper delete_area(reply.area); 9419e9417d2Sbeveloper } else { 9429e9417d2Sbeveloper for (int32 i = 0; i < reply.count; i++) { 9439e9417d2Sbeveloper out_live_nodes[i] = reply.live_info[i]; 9449e9417d2Sbeveloper } 9459e9417d2Sbeveloper } 9469e9417d2Sbeveloper *io_total_count = reply.count; 9479e9417d2Sbeveloper 9489e9417d2Sbeveloper return B_OK; 9499e9417d2Sbeveloper } 9509e9417d2Sbeveloper 95152a38012Sejakowatz 95252a38012Sejakowatz status_t 95352a38012Sejakowatz BMediaRoster::GetFreeInputsFor(const media_node & node, 95452a38012Sejakowatz media_input * out_free_inputs, 95552a38012Sejakowatz int32 buf_num_inputs, 95652a38012Sejakowatz int32 * out_total_count, 95752a38012Sejakowatz media_type filter_type) 95852a38012Sejakowatz { 9593620737cSbeveloper CALLED(); 9609e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_CONSUMER) == 0) 9613620737cSbeveloper return B_MEDIA_BAD_NODE; 9623620737cSbeveloper if (out_free_inputs == NULL || out_total_count == NULL) 9633620737cSbeveloper return B_BAD_VALUE; 9643620737cSbeveloper 9653620737cSbeveloper Stack<media_input> stack; 9663620737cSbeveloper media_input *input; 9673620737cSbeveloper status_t rv; 9683620737cSbeveloper 969*1299bfb2Sbeveloper *out_total_count = 0; 970*1299bfb2Sbeveloper 9713620737cSbeveloper rv = GetAllInputs(node, &stack); 9723620737cSbeveloper if (B_OK != rv) 9733620737cSbeveloper return rv; 9743620737cSbeveloper 9753620737cSbeveloper for (int32 i = 0; stack.GetPointerAt(i, &input); i++) { 9763620737cSbeveloper if (filter_type != B_MEDIA_UNKNOWN_TYPE && filter_type != input->format.type) 9773620737cSbeveloper continue; // media_type used, but doesn't match 9783620737cSbeveloper if (input->source != media_source::null) 9793620737cSbeveloper continue; // consumer source already connected 9803620737cSbeveloper out_free_inputs[i] = *input; 9813620737cSbeveloper *out_total_count += 1; 9823620737cSbeveloper buf_num_inputs -= 1; 9833620737cSbeveloper if (buf_num_inputs == 0) 9843620737cSbeveloper break; 9853620737cSbeveloper } 9863620737cSbeveloper 9873620737cSbeveloper PublishInputs(node, &stack); 9883620737cSbeveloper return B_OK; 98952a38012Sejakowatz } 99052a38012Sejakowatz 99152a38012Sejakowatz 99252a38012Sejakowatz status_t 99352a38012Sejakowatz BMediaRoster::GetConnectedInputsFor(const media_node & node, 99452a38012Sejakowatz media_input * out_active_inputs, 99552a38012Sejakowatz int32 buf_num_inputs, 99652a38012Sejakowatz int32 * out_total_count) 99752a38012Sejakowatz { 9983620737cSbeveloper CALLED(); 9999e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_CONSUMER) == 0) 10003620737cSbeveloper return B_MEDIA_BAD_NODE; 10013620737cSbeveloper if (out_active_inputs == NULL || out_total_count == NULL) 10023620737cSbeveloper return B_BAD_VALUE; 10033620737cSbeveloper 10043620737cSbeveloper Stack<media_input> stack; 10053620737cSbeveloper media_input *input; 10063620737cSbeveloper status_t rv; 10073620737cSbeveloper 1008*1299bfb2Sbeveloper *out_total_count = 0; 1009*1299bfb2Sbeveloper 10103620737cSbeveloper rv = GetAllInputs(node, &stack); 10113620737cSbeveloper if (B_OK != rv) 10123620737cSbeveloper return rv; 10133620737cSbeveloper 10143620737cSbeveloper for (int32 i = 0; stack.GetPointerAt(i, &input); i++) { 10153620737cSbeveloper if (input->source == media_source::null) 10163620737cSbeveloper continue; // consumer source not connected 10173620737cSbeveloper out_active_inputs[i] = *input; 10183620737cSbeveloper *out_total_count += 1; 10193620737cSbeveloper buf_num_inputs -= 1; 10203620737cSbeveloper if (buf_num_inputs == 0) 10213620737cSbeveloper break; 10223620737cSbeveloper } 10233620737cSbeveloper 10243620737cSbeveloper PublishInputs(node, &stack); 10253620737cSbeveloper return B_OK; 102652a38012Sejakowatz } 102752a38012Sejakowatz 102852a38012Sejakowatz 102952a38012Sejakowatz status_t 103052a38012Sejakowatz BMediaRoster::GetAllInputsFor(const media_node & node, 103152a38012Sejakowatz media_input * out_inputs, 103252a38012Sejakowatz int32 buf_num_inputs, 103352a38012Sejakowatz int32 * out_total_count) 103452a38012Sejakowatz { 103552a38012Sejakowatz CALLED(); 10369e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_CONSUMER) == 0) 103752a38012Sejakowatz return B_MEDIA_BAD_NODE; 103852a38012Sejakowatz if (out_inputs == NULL || out_total_count == NULL) 103952a38012Sejakowatz return B_BAD_VALUE; 104052a38012Sejakowatz 10413620737cSbeveloper Stack<media_input> stack; 10423620737cSbeveloper media_input *input; 104352a38012Sejakowatz status_t rv; 104452a38012Sejakowatz 1045*1299bfb2Sbeveloper *out_total_count = 0; 1046*1299bfb2Sbeveloper 10473620737cSbeveloper rv = GetAllInputs(node, &stack); 10483620737cSbeveloper if (B_OK != rv) 10493620737cSbeveloper return rv; 10503620737cSbeveloper 10513620737cSbeveloper for (int32 i = 0; stack.GetPointerAt(i, &input); i++) { 10523620737cSbeveloper out_inputs[i] = *input; 105352a38012Sejakowatz *out_total_count += 1; 10543620737cSbeveloper buf_num_inputs -= 1; 10553620737cSbeveloper if (buf_num_inputs == 0) 10563620737cSbeveloper break; 105752a38012Sejakowatz } 105852a38012Sejakowatz 10593620737cSbeveloper PublishInputs(node, &stack); 10603620737cSbeveloper return B_OK; 106152a38012Sejakowatz } 106252a38012Sejakowatz 106352a38012Sejakowatz 106452a38012Sejakowatz status_t 106552a38012Sejakowatz BMediaRoster::GetFreeOutputsFor(const media_node & node, 106652a38012Sejakowatz media_output * out_free_outputs, 106752a38012Sejakowatz int32 buf_num_outputs, 106852a38012Sejakowatz int32 * out_total_count, 106952a38012Sejakowatz media_type filter_type) 107052a38012Sejakowatz { 10713620737cSbeveloper CALLED(); 10729e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_PRODUCER) == 0) 10733620737cSbeveloper return B_MEDIA_BAD_NODE; 10743620737cSbeveloper if (out_free_outputs == NULL || out_total_count == NULL) 10753620737cSbeveloper return B_BAD_VALUE; 10763620737cSbeveloper 10773620737cSbeveloper Stack<media_output> stack; 10783620737cSbeveloper media_output *output; 10793620737cSbeveloper status_t rv; 10803620737cSbeveloper 1081*1299bfb2Sbeveloper *out_total_count = 0; 1082*1299bfb2Sbeveloper 10833620737cSbeveloper rv = GetAllOutputs(node, &stack); 10843620737cSbeveloper if (B_OK != rv) 10853620737cSbeveloper return rv; 10863620737cSbeveloper 10873620737cSbeveloper for (int32 i = 0; stack.GetPointerAt(i, &output); i++) { 10883620737cSbeveloper if (filter_type != B_MEDIA_UNKNOWN_TYPE && filter_type != output->format.type) 10893620737cSbeveloper continue; // media_type used, but doesn't match 10903620737cSbeveloper if (output->destination != media_destination::null) 10913620737cSbeveloper continue; // producer destination already connected 10923620737cSbeveloper out_free_outputs[i] = *output; 10933620737cSbeveloper *out_total_count += 1; 10943620737cSbeveloper buf_num_outputs -= 1; 10953620737cSbeveloper if (buf_num_outputs == 0) 10963620737cSbeveloper break; 10973620737cSbeveloper } 10983620737cSbeveloper 10993620737cSbeveloper PublishOutputs(node, &stack); 11003620737cSbeveloper return B_OK; 110152a38012Sejakowatz } 110252a38012Sejakowatz 110352a38012Sejakowatz 110452a38012Sejakowatz status_t 110552a38012Sejakowatz BMediaRoster::GetConnectedOutputsFor(const media_node & node, 110652a38012Sejakowatz media_output * out_active_outputs, 110752a38012Sejakowatz int32 buf_num_outputs, 110852a38012Sejakowatz int32 * out_total_count) 110952a38012Sejakowatz { 11103620737cSbeveloper CALLED(); 11119e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_PRODUCER) == 0) 11123620737cSbeveloper return B_MEDIA_BAD_NODE; 11133620737cSbeveloper if (out_active_outputs == NULL || out_total_count == NULL) 11143620737cSbeveloper return B_BAD_VALUE; 11153620737cSbeveloper 11163620737cSbeveloper Stack<media_output> stack; 11173620737cSbeveloper media_output *output; 11183620737cSbeveloper status_t rv; 11193620737cSbeveloper 1120*1299bfb2Sbeveloper *out_total_count = 0; 1121*1299bfb2Sbeveloper 11223620737cSbeveloper rv = GetAllOutputs(node, &stack); 11233620737cSbeveloper if (B_OK != rv) 11243620737cSbeveloper return rv; 11253620737cSbeveloper 11263620737cSbeveloper for (int32 i = 0; stack.GetPointerAt(i, &output); i++) { 11273620737cSbeveloper if (output->destination == media_destination::null) 11283620737cSbeveloper continue; // producer destination not connected 11293620737cSbeveloper out_active_outputs[i] = *output; 11303620737cSbeveloper *out_total_count += 1; 11313620737cSbeveloper buf_num_outputs -= 1; 11323620737cSbeveloper if (buf_num_outputs == 0) 11333620737cSbeveloper break; 11343620737cSbeveloper } 11353620737cSbeveloper 11363620737cSbeveloper PublishOutputs(node, &stack); 11373620737cSbeveloper return B_OK; 113852a38012Sejakowatz } 113952a38012Sejakowatz 114052a38012Sejakowatz 114152a38012Sejakowatz status_t 114252a38012Sejakowatz BMediaRoster::GetAllOutputsFor(const media_node & node, 114352a38012Sejakowatz media_output * out_outputs, 114452a38012Sejakowatz int32 buf_num_outputs, 114552a38012Sejakowatz int32 * out_total_count) 114652a38012Sejakowatz { 114752a38012Sejakowatz CALLED(); 11489e9417d2Sbeveloper if (node.node <= 0 || (node.kind & B_BUFFER_PRODUCER) == 0) 114952a38012Sejakowatz return B_MEDIA_BAD_NODE; 115052a38012Sejakowatz if (out_outputs == NULL || out_total_count == NULL) 115152a38012Sejakowatz return B_BAD_VALUE; 115252a38012Sejakowatz 11533620737cSbeveloper Stack<media_output> stack; 11543620737cSbeveloper media_output *output; 115552a38012Sejakowatz status_t rv; 115652a38012Sejakowatz 1157*1299bfb2Sbeveloper *out_total_count = 0; 1158*1299bfb2Sbeveloper 11593620737cSbeveloper rv = GetAllOutputs(node, &stack); 11603620737cSbeveloper if (B_OK != rv) 11613620737cSbeveloper return rv; 11623620737cSbeveloper 11633620737cSbeveloper for (int32 i = 0; stack.GetPointerAt(i, &output); i++) { 11643620737cSbeveloper out_outputs[i] = *output; 116552a38012Sejakowatz *out_total_count += 1; 11663620737cSbeveloper buf_num_outputs -= 1; 11673620737cSbeveloper if (buf_num_outputs == 0) 11683620737cSbeveloper break; 116952a38012Sejakowatz } 117052a38012Sejakowatz 11713620737cSbeveloper PublishOutputs(node, &stack); 11723620737cSbeveloper return B_OK; 117352a38012Sejakowatz } 117452a38012Sejakowatz 117552a38012Sejakowatz 117652a38012Sejakowatz status_t 117752a38012Sejakowatz BMediaRoster::StartWatching(const BMessenger & where) 117852a38012Sejakowatz { 1179eae26d3dSbeveloper CALLED(); 1180eae26d3dSbeveloper if (!where.IsValid()) { 1181eae26d3dSbeveloper TRACE("BMediaRoster::StartWatching: messenger invalid!\n"); 1182eae26d3dSbeveloper return B_BAD_VALUE; 1183eae26d3dSbeveloper } 1184f4925104Sbeveloper return BPrivate::media::notifications::Register(where, media_node::null, B_MEDIA_WILDCARD); 118552a38012Sejakowatz } 118652a38012Sejakowatz 118752a38012Sejakowatz 118852a38012Sejakowatz status_t 118952a38012Sejakowatz BMediaRoster::StartWatching(const BMessenger & where, 119052a38012Sejakowatz int32 notificationType) 119152a38012Sejakowatz { 1192eae26d3dSbeveloper CALLED(); 1193eae26d3dSbeveloper if (!where.IsValid()) { 1194eae26d3dSbeveloper TRACE("BMediaRoster::StartWatching: messenger invalid!\n"); 1195eae26d3dSbeveloper return B_BAD_VALUE; 1196eae26d3dSbeveloper } 1197f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(false, notificationType)) { 1198eae26d3dSbeveloper TRACE("BMediaRoster::StartWatching: notificationType invalid!\n"); 1199eae26d3dSbeveloper return B_BAD_VALUE; 1200eae26d3dSbeveloper } 1201f4925104Sbeveloper return BPrivate::media::notifications::Register(where, media_node::null, notificationType); 120252a38012Sejakowatz } 120352a38012Sejakowatz 120452a38012Sejakowatz 120552a38012Sejakowatz status_t 120652a38012Sejakowatz BMediaRoster::StartWatching(const BMessenger & where, 120752a38012Sejakowatz const media_node & node, 120852a38012Sejakowatz int32 notificationType) 120952a38012Sejakowatz { 1210eae26d3dSbeveloper CALLED(); 1211eae26d3dSbeveloper if (!where.IsValid()) { 1212eae26d3dSbeveloper TRACE("BMediaRoster::StartWatching: messenger invalid!\n"); 1213eae26d3dSbeveloper return B_BAD_VALUE; 1214eae26d3dSbeveloper } 12159e9417d2Sbeveloper if (node.node <= 0) { 1216eae26d3dSbeveloper TRACE("BMediaRoster::StartWatching: node invalid!\n"); 1217eae26d3dSbeveloper return B_MEDIA_BAD_NODE; 1218eae26d3dSbeveloper } 1219f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(true, notificationType)) { 1220eae26d3dSbeveloper TRACE("BMediaRoster::StartWatching: notificationType invalid!\n"); 1221eae26d3dSbeveloper return B_BAD_VALUE; 1222eae26d3dSbeveloper } 1223f4925104Sbeveloper return BPrivate::media::notifications::Register(where, node, notificationType); 122452a38012Sejakowatz } 122552a38012Sejakowatz 122652a38012Sejakowatz 122752a38012Sejakowatz status_t 122852a38012Sejakowatz BMediaRoster::StopWatching(const BMessenger & where) 122952a38012Sejakowatz { 1230eae26d3dSbeveloper CALLED(); 1231eae26d3dSbeveloper // messenger may already be invalid, so we don't check this 1232f4925104Sbeveloper return BPrivate::media::notifications::Unregister(where, media_node::null, B_MEDIA_WILDCARD); 123352a38012Sejakowatz } 123452a38012Sejakowatz 123552a38012Sejakowatz 123652a38012Sejakowatz status_t 123752a38012Sejakowatz BMediaRoster::StopWatching(const BMessenger & where, 123852a38012Sejakowatz int32 notificationType) 123952a38012Sejakowatz { 1240eae26d3dSbeveloper CALLED(); 1241eae26d3dSbeveloper // messenger may already be invalid, so we don't check this 1242f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(false, notificationType)) { 1243eae26d3dSbeveloper TRACE("BMediaRoster::StopWatching: notificationType invalid!\n"); 1244eae26d3dSbeveloper return B_BAD_VALUE; 1245eae26d3dSbeveloper } 1246f4925104Sbeveloper return BPrivate::media::notifications::Unregister(where, media_node::null, notificationType); 124752a38012Sejakowatz } 124852a38012Sejakowatz 124952a38012Sejakowatz 125052a38012Sejakowatz status_t 125152a38012Sejakowatz BMediaRoster::StopWatching(const BMessenger & where, 125252a38012Sejakowatz const media_node & node, 125352a38012Sejakowatz int32 notificationType) 125452a38012Sejakowatz { 1255eae26d3dSbeveloper CALLED(); 1256eae26d3dSbeveloper // messenger may already be invalid, so we don't check this 12579e9417d2Sbeveloper if (node.node <= 0) { 1258eae26d3dSbeveloper TRACE("BMediaRoster::StopWatching: node invalid!\n"); 1259eae26d3dSbeveloper return B_MEDIA_BAD_NODE; 1260eae26d3dSbeveloper } 1261f4925104Sbeveloper if (false == BPrivate::media::notifications::IsValidNotificationRequest(true, notificationType)) { 1262eae26d3dSbeveloper TRACE("BMediaRoster::StopWatching: notificationType invalid!\n"); 1263eae26d3dSbeveloper return B_BAD_VALUE; 1264eae26d3dSbeveloper } 1265f4925104Sbeveloper return BPrivate::media::notifications::Unregister(where, node, notificationType); 126652a38012Sejakowatz } 126752a38012Sejakowatz 126852a38012Sejakowatz 126952a38012Sejakowatz status_t 127052a38012Sejakowatz BMediaRoster::RegisterNode(BMediaNode * node) 127152a38012Sejakowatz { 127252a38012Sejakowatz CALLED(); 127352a38012Sejakowatz if (node == NULL) 127452a38012Sejakowatz return B_BAD_VALUE; 127552a38012Sejakowatz 12763620737cSbeveloper status_t rv; 12779e9417d2Sbeveloper BMediaAddOn *addon; 12789e9417d2Sbeveloper int32 addon_flavor_id; 12799e9417d2Sbeveloper media_addon_id addon_id; 12803620737cSbeveloper 12819e9417d2Sbeveloper addon_flavor_id = 0; 12829e9417d2Sbeveloper addon = node->AddOn(&addon_flavor_id); 12839e9417d2Sbeveloper addon_id = addon ? addon->AddonID() : -1; 128452a38012Sejakowatz 12859e9417d2Sbeveloper server_register_node_request request; 12869e9417d2Sbeveloper server_register_node_reply reply; 12879e9417d2Sbeveloper 12889e9417d2Sbeveloper request.addon_id = addon_id; 12899e9417d2Sbeveloper request.addon_flavor_id = addon_flavor_id; 12909e9417d2Sbeveloper strcpy(request.name, node->Name()); 12919e9417d2Sbeveloper request.kinds = node->Kinds(); 12929e9417d2Sbeveloper request.port = node->ControlPort(); 12939e9417d2Sbeveloper request.team = team; 12949e9417d2Sbeveloper 1295*1299bfb2Sbeveloper TRACE("BMediaRoster::RegisterNode: sending SERVER_REGISTER_NODE: port %ld, kinds %#Lx, team %ld, name '%s'\n", request.port, request.kinds, request.team, request.name); 1296*1299bfb2Sbeveloper 12979e9417d2Sbeveloper rv = QueryServer(SERVER_REGISTER_NODE, &request, sizeof(request), &reply, sizeof(reply)); 12989e9417d2Sbeveloper if (rv != B_OK) { 12999e9417d2Sbeveloper TRACE("BMediaRoster::RegisterNode: failed to register node %s (error %#lx)\n", node->Name(), rv); 13009e9417d2Sbeveloper return rv; 13019e9417d2Sbeveloper } 13029e9417d2Sbeveloper 13039e9417d2Sbeveloper // we are a friend class of BMediaNode and initilize this member variable 13049e9417d2Sbeveloper node->fNodeID = reply.nodeid; 13059e9417d2Sbeveloper ASSERT(reply.nodeid == node->Node().node); 13069e9417d2Sbeveloper ASSERT(reply.nodeid == node->ID()); 13079e9417d2Sbeveloper 1308*1299bfb2Sbeveloper TRACE("BMediaRoster::RegisterNode: before callback: port %ld, name '%s'\n", node->ControlPort(), node->Name()); 1309*1299bfb2Sbeveloper 13109e9417d2Sbeveloper // call the callback 13119e9417d2Sbeveloper node->NodeRegistered(); 13123620737cSbeveloper 1313*1299bfb2Sbeveloper TRACE("BMediaRoster::RegisterNode: after callback: port %ld, name '%s'\n", node->ControlPort(), node->Name()); 1314*1299bfb2Sbeveloper 13153620737cSbeveloper // register existing inputs and outputs with the 13163620737cSbeveloper // media_server, this allows GetLiveNodes() to work 13173620737cSbeveloper // with created, but unconnected nodes. 13183620737cSbeveloper if (node->Kinds() & B_BUFFER_PRODUCER) { 13193620737cSbeveloper Stack<media_output> stack; 13203620737cSbeveloper if (B_OK == GetAllOutputs(node->Node(), &stack)) 13213620737cSbeveloper PublishOutputs(node->Node(), &stack); 13223620737cSbeveloper } else if (node->Kinds() & B_BUFFER_CONSUMER) { 13233620737cSbeveloper Stack<media_input> stack; 13243620737cSbeveloper if (B_OK == GetAllInputs(node->Node(), &stack)) 13253620737cSbeveloper PublishInputs(node->Node(), &stack); 13263620737cSbeveloper } 1327*1299bfb2Sbeveloper BPrivate::media::notifications::NodesCreated(&reply.nodeid, 1); 1328*1299bfb2Sbeveloper 13293620737cSbeveloper 13309e9417d2Sbeveloper TRACE("BMediaRoster::RegisterNode: registered node %s, id %ld, addon %ld, flavor %ld\n", node->Name(), node->ID(), addon_id, addon_flavor_id); 13319e9417d2Sbeveloper 13329e9417d2Sbeveloper return B_OK; 133352a38012Sejakowatz } 133452a38012Sejakowatz 133552a38012Sejakowatz 133652a38012Sejakowatz status_t 133752a38012Sejakowatz BMediaRoster::UnregisterNode(BMediaNode * node) 133852a38012Sejakowatz { 13399e9417d2Sbeveloper CALLED(); 13409e9417d2Sbeveloper if (node == NULL) 13419e9417d2Sbeveloper return B_BAD_VALUE; 13429e9417d2Sbeveloper 13439e9417d2Sbeveloper if (node->fRefCount != 0) { 13449e9417d2Sbeveloper TRACE("BMediaRoster::UnregisterNode: Warning node %s has local reference count of %ld\n", node->Name(), node->fRefCount); 13459e9417d2Sbeveloper // no return here, we continue and unregister! 13469e9417d2Sbeveloper } 13479e9417d2Sbeveloper if (node->ID() == -2) { 13489e9417d2Sbeveloper TRACE("BMediaRoster::UnregisterNode: Warning node %s already unregistered\n", node->Name()); 13499e9417d2Sbeveloper return B_OK; 13509e9417d2Sbeveloper } 13519e9417d2Sbeveloper 13529e9417d2Sbeveloper server_unregister_node_request request; 13539e9417d2Sbeveloper server_unregister_node_reply reply; 13549e9417d2Sbeveloper status_t rv; 13559e9417d2Sbeveloper 13569e9417d2Sbeveloper request.nodeid = node->ID(); 13579e9417d2Sbeveloper request.team = team; 13589e9417d2Sbeveloper 1359*1299bfb2Sbeveloper // send a notification 1360*1299bfb2Sbeveloper BPrivate::media::notifications::NodesDeleted(&request.nodeid, 1); 1361*1299bfb2Sbeveloper 13629e9417d2Sbeveloper rv = QueryServer(SERVER_UNREGISTER_NODE, &request, sizeof(request), &reply, sizeof(reply)); 13639e9417d2Sbeveloper if (rv != B_OK) { 13649e9417d2Sbeveloper TRACE("BMediaRoster::UnregisterNode: failed to unregister node %s (error %#lx)\n", node->Name(), rv); 13659e9417d2Sbeveloper return rv; 13669e9417d2Sbeveloper } 13679e9417d2Sbeveloper 13689e9417d2Sbeveloper if (reply.addon_id != -1) 13699e9417d2Sbeveloper _DormantNodeManager->PutAddon(reply.addon_id); 13709e9417d2Sbeveloper 1371*1299bfb2Sbeveloper // we are a friend class of BMediaNode and invalidate this member variable 13729e9417d2Sbeveloper node->fNodeID = -2; 13739e9417d2Sbeveloper 13749e9417d2Sbeveloper return B_OK; 137552a38012Sejakowatz } 137652a38012Sejakowatz 137752a38012Sejakowatz 137852a38012Sejakowatz // thread safe for multiple calls to Roster() 137952a38012Sejakowatz /* static */ BMediaRoster * 138052a38012Sejakowatz BMediaRoster::Roster(status_t* out_error) 138152a38012Sejakowatz { 138252a38012Sejakowatz CALLED(); 138352a38012Sejakowatz static BLocker locker("BMediaRoster::Roster locker"); 138452a38012Sejakowatz locker.Lock(); 138552a38012Sejakowatz if (_sDefault == NULL) { 138652a38012Sejakowatz _sDefault = new BMediaRoster(); 138752a38012Sejakowatz if (out_error != NULL) 138852a38012Sejakowatz *out_error = B_OK; 138952a38012Sejakowatz } else { 139052a38012Sejakowatz if (out_error != NULL) 139152a38012Sejakowatz *out_error = B_OK; 139252a38012Sejakowatz } 139352a38012Sejakowatz locker.Unlock(); 139452a38012Sejakowatz return _sDefault; 139552a38012Sejakowatz } 139652a38012Sejakowatz 139752a38012Sejakowatz 139852a38012Sejakowatz // won't create it if there isn't one 139952a38012Sejakowatz // not thread safe if you call Roster() at the same time 140052a38012Sejakowatz /* static */ BMediaRoster * 140152a38012Sejakowatz BMediaRoster::CurrentRoster() 140252a38012Sejakowatz { 140352a38012Sejakowatz CALLED(); 140452a38012Sejakowatz return _sDefault; 140552a38012Sejakowatz } 140652a38012Sejakowatz 140752a38012Sejakowatz 140852a38012Sejakowatz status_t 140952a38012Sejakowatz BMediaRoster::SetTimeSourceFor(media_node_id node, 141052a38012Sejakowatz media_node_id time_source) 141152a38012Sejakowatz { 141252a38012Sejakowatz UNIMPLEMENTED(); 141352a38012Sejakowatz return B_ERROR; 141452a38012Sejakowatz } 141552a38012Sejakowatz 141652a38012Sejakowatz 141752a38012Sejakowatz status_t 141852a38012Sejakowatz BMediaRoster::GetParameterWebFor(const media_node & node, 141952a38012Sejakowatz BParameterWeb ** out_web) 142052a38012Sejakowatz { 142152a38012Sejakowatz UNIMPLEMENTED(); 142252a38012Sejakowatz return B_ERROR; 142352a38012Sejakowatz } 142452a38012Sejakowatz 142552a38012Sejakowatz 142652a38012Sejakowatz status_t 142752a38012Sejakowatz BMediaRoster::StartControlPanel(const media_node & node, 142852a38012Sejakowatz BMessenger * out_messenger) 142952a38012Sejakowatz { 143052a38012Sejakowatz UNIMPLEMENTED(); 143152a38012Sejakowatz return B_ERROR; 143252a38012Sejakowatz } 143352a38012Sejakowatz 143452a38012Sejakowatz 143552a38012Sejakowatz status_t 143652a38012Sejakowatz BMediaRoster::GetDormantNodes(dormant_node_info * out_info, 143752a38012Sejakowatz int32 * io_count, 143852a38012Sejakowatz const media_format * has_input /* = NULL */, 143952a38012Sejakowatz const media_format * has_output /* = NULL */, 144052a38012Sejakowatz const char * name /* = NULL */, 144152a38012Sejakowatz uint64 require_kinds /* = NULL */, 144252a38012Sejakowatz uint64 deny_kinds /* = NULL */) 144352a38012Sejakowatz { 144452a38012Sejakowatz CALLED(); 144552a38012Sejakowatz if (out_info == NULL) 144652a38012Sejakowatz return B_BAD_VALUE; 144752a38012Sejakowatz if (io_count == NULL) 144852a38012Sejakowatz return B_BAD_VALUE; 144952a38012Sejakowatz if (*io_count <= 0) 145052a38012Sejakowatz return B_BAD_VALUE; 145152a38012Sejakowatz 145252a38012Sejakowatz xfer_server_get_dormant_nodes msg; 145352a38012Sejakowatz port_id port; 145452a38012Sejakowatz status_t rv; 145552a38012Sejakowatz 145652a38012Sejakowatz port = find_port("media_server port"); 145752a38012Sejakowatz if (port <= B_OK) 145852a38012Sejakowatz return B_ERROR; 145952a38012Sejakowatz 146052a38012Sejakowatz msg.maxcount = *io_count; 146152a38012Sejakowatz msg.has_input = (bool) has_input; 146252a38012Sejakowatz if (has_input) 146352a38012Sejakowatz msg.inputformat = *has_input; // XXX we should not make a flat copy of media_format 146452a38012Sejakowatz msg.has_output = (bool) has_output; 146552a38012Sejakowatz if (has_output) 146652a38012Sejakowatz msg.outputformat = *has_output;; // XXX we should not make a flat copy of media_format 146752a38012Sejakowatz msg.has_name = (bool) name; 146852a38012Sejakowatz if (name) { 14699e9417d2Sbeveloper int len = strlen(name); 14709e9417d2Sbeveloper len = min_c(len, (int)sizeof(msg.name) - 1); 147152a38012Sejakowatz memcpy(msg.name, name, len); 147252a38012Sejakowatz msg.name[len] = 0; 147352a38012Sejakowatz } 147452a38012Sejakowatz msg.require_kinds = require_kinds; 147552a38012Sejakowatz msg.deny_kinds = deny_kinds; 147652a38012Sejakowatz msg.reply_port = _PortPool->GetPort(); 147752a38012Sejakowatz 147852a38012Sejakowatz rv = write_port(port, SERVER_GET_DORMANT_NODES, &msg, sizeof(msg)); 147952a38012Sejakowatz if (rv != B_OK) { 148052a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 148152a38012Sejakowatz return rv; 148252a38012Sejakowatz } 148352a38012Sejakowatz 148452a38012Sejakowatz xfer_server_get_dormant_nodes_reply reply; 148552a38012Sejakowatz int32 code; 148652a38012Sejakowatz 148752a38012Sejakowatz rv = read_port(msg.reply_port, &code, &reply, sizeof(reply)); 148852a38012Sejakowatz if (rv < B_OK) { 148952a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 149052a38012Sejakowatz return rv; 149152a38012Sejakowatz } 149252a38012Sejakowatz 149352a38012Sejakowatz *io_count = reply.count; 149452a38012Sejakowatz 149552a38012Sejakowatz if (*io_count > 0) { 149652a38012Sejakowatz rv = read_port(msg.reply_port, &code, out_info, *io_count * sizeof(dormant_node_info)); 149752a38012Sejakowatz if (rv < B_OK) 149852a38012Sejakowatz reply.result = rv; 149952a38012Sejakowatz } 150052a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 150152a38012Sejakowatz 150252a38012Sejakowatz return reply.result; 150352a38012Sejakowatz } 150452a38012Sejakowatz 150552a38012Sejakowatz 150652a38012Sejakowatz status_t 150752a38012Sejakowatz BMediaRoster::InstantiateDormantNode(const dormant_node_info & in_info, 150852a38012Sejakowatz media_node * out_node, 150952a38012Sejakowatz uint32 flags /* currently B_FLAVOR_IS_GLOBAL or B_FLAVOR_IS_LOCAL */ ) 151052a38012Sejakowatz { 151152a38012Sejakowatz CALLED(); 151252a38012Sejakowatz if ((flags & (B_FLAVOR_IS_GLOBAL | B_FLAVOR_IS_LOCAL)) == 0) { 151352a38012Sejakowatz printf("Error: BMediaRoster::InstantiateDormantNode called without flags\n"); 151452a38012Sejakowatz return B_BAD_VALUE; 151552a38012Sejakowatz } 151652a38012Sejakowatz if (out_node == 0) 151752a38012Sejakowatz return B_BAD_VALUE; 151852a38012Sejakowatz 151952a38012Sejakowatz // XXX we should not trust the values passed in by the user, 152052a38012Sejakowatz // XXX and ask the server to determine where to insta 152152a38012Sejakowatz 1522e0207628Sbeveloper 1523e0207628Sbeveloper // XXX SOMETHING IS VERY WRONG HERE 1524e0207628Sbeveloper // if ((in_info.flavor_flags & B_FLAVOR_IS_GLOBAL) == 0 && (flags & B_FLAVOR_IS_LOCAL)) { 1525e0207628Sbeveloper if (flags & B_FLAVOR_IS_LOCAL) { 152652a38012Sejakowatz return InstantiateDormantNode(in_info,out_node); 152752a38012Sejakowatz } 152852a38012Sejakowatz 1529e0207628Sbeveloper // XXX SOMETHING IS VERY WRONG HERE 1530e0207628Sbeveloper // if ((in_info.flavor_flags & B_FLAVOR_IS_GLOBAL) || (flags & B_FLAVOR_IS_GLOBAL)) { 1531e0207628Sbeveloper if (flags & B_FLAVOR_IS_GLOBAL) { 153252a38012Sejakowatz // forward this request into the media_addon_server, 153352a38012Sejakowatz // which in turn will call InstantiateDormantNode() 153452a38012Sejakowatz // to create it there localy 15358c6a6096Sbeveloper addonserver_instantiate_dormant_node_request request; 15368c6a6096Sbeveloper addonserver_instantiate_dormant_node_reply reply; 153752a38012Sejakowatz status_t rv; 1538f4925104Sbeveloper 1539f4925104Sbeveloper request.info = in_info; 1540f4925104Sbeveloper rv = QueryAddonServer(ADDONSERVER_INSTANTIATE_DORMANT_NODE, &request, sizeof(request), &reply, sizeof(reply)); 1541f4925104Sbeveloper if (rv == B_OK) { 154252a38012Sejakowatz *out_node = reply.node; 1543f4925104Sbeveloper } 1544f4925104Sbeveloper return rv; 154552a38012Sejakowatz } 154652a38012Sejakowatz 1547e0207628Sbeveloper // XXX SOMETHING IS VERY WRONG HERE 15485edd2016Sbeveloper printf("Error: BMediaRoster::InstantiateDormantNode addon_id %d, flavor_id %d, flags %#08lx\n", (int)in_info.addon, (int)in_info.flavor_id, flags); 154952a38012Sejakowatz 155052a38012Sejakowatz return B_ERROR; 155152a38012Sejakowatz } 155252a38012Sejakowatz 155352a38012Sejakowatz 155452a38012Sejakowatz status_t 155552a38012Sejakowatz BMediaRoster::InstantiateDormantNode(const dormant_node_info & in_info, 155652a38012Sejakowatz media_node * out_node) 155752a38012Sejakowatz { 155816b7eea4Sbeveloper CALLED(); 155952a38012Sejakowatz 156052a38012Sejakowatz // to instantiate a dormant node in the current address space, we need to 156152a38012Sejakowatz // either load the add-on from file and create a new BMediaAddOn class, or 156252a38012Sejakowatz // reuse the cached BMediaAddOn from a previous call 156352a38012Sejakowatz // call BMediaAddOn::InstantiateNodeFor() 156452a38012Sejakowatz // and cache the BMediaAddOn after that for later reuse. 156552a38012Sejakowatz // BeOS R5 does not seem to delete it when the application quits 156652a38012Sejakowatz // if B_FLAVOR_IS_GLOBAL, we need to use the BMediaAddOn object that 156752a38012Sejakowatz // resides in the media_addon_server 156852a38012Sejakowatz 156952a38012Sejakowatz // RegisterNode() is called automatically for nodes instantiated from add-ons 157052a38012Sejakowatz 157116b7eea4Sbeveloper //XXX TEST! 157216b7eea4Sbeveloper BMediaAddOn *addon; 157316b7eea4Sbeveloper BMediaNode *node; 157416b7eea4Sbeveloper BMessage config; 157516b7eea4Sbeveloper status_t out_error; 15768c6a6096Sbeveloper status_t rv; 157716b7eea4Sbeveloper addon = _DormantNodeManager->GetAddon(in_info.addon); 157816b7eea4Sbeveloper if (!addon) { 157916b7eea4Sbeveloper printf("BMediaRoster::InstantiateDormantNode: GetAddon failed\n"); 158052a38012Sejakowatz return B_ERROR; 158152a38012Sejakowatz } 1582*1299bfb2Sbeveloper flavor_info temp; // XXX fix this! 1583*1299bfb2Sbeveloper temp.name = "XXX flavor_info name"; 1584*1299bfb2Sbeveloper temp.info = "XXX flavor_info info"; 158516b7eea4Sbeveloper temp.internal_id = in_info.flavor_id; 158616b7eea4Sbeveloper node = addon->InstantiateNodeFor(&temp, &config, &out_error); 158716b7eea4Sbeveloper if (!node) { 158816b7eea4Sbeveloper printf("BMediaRoster::InstantiateDormantNode: InstantiateNodeFor failed\n"); 158916b7eea4Sbeveloper _DormantNodeManager->PutAddon(in_info.addon); 159016b7eea4Sbeveloper return B_ERROR; 159116b7eea4Sbeveloper } 15928c6a6096Sbeveloper rv = RegisterNode(node); 15938c6a6096Sbeveloper if (rv != B_OK) { 15948c6a6096Sbeveloper printf("BMediaRoster::InstantiateDormantNode: RegisterNode failed\n"); 15958c6a6096Sbeveloper delete node; 15968c6a6096Sbeveloper _DormantNodeManager->PutAddon(in_info.addon); 15978c6a6096Sbeveloper return B_ERROR; 15988c6a6096Sbeveloper } 15998c6a6096Sbeveloper 16008c6a6096Sbeveloper // XXX we must remember in_info.addon and call 16018c6a6096Sbeveloper // XXX _DormantNodeManager->PutAddon when the 16028c6a6096Sbeveloper // XXX node is unregistered 16039e9417d2Sbeveloper // should be handled by RegisterNode() and UnegisterNode() now 16048c6a6096Sbeveloper 160516b7eea4Sbeveloper *out_node = node->Node(); 160616b7eea4Sbeveloper return B_OK; 160716b7eea4Sbeveloper } 160852a38012Sejakowatz 160952a38012Sejakowatz 161052a38012Sejakowatz status_t 161152a38012Sejakowatz BMediaRoster::GetDormantNodeFor(const media_node & node, 161252a38012Sejakowatz dormant_node_info * out_info) 161352a38012Sejakowatz { 161485dfab2bSbeveloper CALLED(); 161585dfab2bSbeveloper if (out_info == NULL) 161685dfab2bSbeveloper return B_BAD_VALUE; 161785dfab2bSbeveloper if (node.node <= 0) 161885dfab2bSbeveloper return B_MEDIA_BAD_NODE; 161952a38012Sejakowatz 162085dfab2bSbeveloper server_get_dormant_node_for_request request; 162185dfab2bSbeveloper server_get_dormant_node_for_reply reply; 162285dfab2bSbeveloper status_t rv; 162385dfab2bSbeveloper 162485dfab2bSbeveloper request.node = node; 162585dfab2bSbeveloper 1626*1299bfb2Sbeveloper rv = QueryServer(SERVER_GET_DORMANT_NODE_FOR, &request, sizeof(request), &reply, sizeof(reply)); 162785dfab2bSbeveloper if (rv != B_OK) 162885dfab2bSbeveloper return rv; 162985dfab2bSbeveloper 163085dfab2bSbeveloper *out_info = reply.node_info; 163185dfab2bSbeveloper return B_OK; 163252a38012Sejakowatz } 163352a38012Sejakowatz 163452a38012Sejakowatz 163552a38012Sejakowatz status_t 163652a38012Sejakowatz BMediaRoster::GetDormantFlavorInfoFor(const dormant_node_info & in_dormant, 163752a38012Sejakowatz dormant_flavor_info * out_flavor) 163852a38012Sejakowatz { 163952a38012Sejakowatz CALLED(); 164052a38012Sejakowatz 164152a38012Sejakowatz xfer_server_get_dormant_flavor_info msg; 164252a38012Sejakowatz xfer_server_get_dormant_flavor_info_reply *reply; 164352a38012Sejakowatz port_id port; 164452a38012Sejakowatz status_t rv; 164552a38012Sejakowatz int32 code; 164652a38012Sejakowatz 164752a38012Sejakowatz port = find_port("media_server port"); 164852a38012Sejakowatz if (port <= B_OK) 164952a38012Sejakowatz return B_ERROR; 165052a38012Sejakowatz 165152a38012Sejakowatz reply = (xfer_server_get_dormant_flavor_info_reply *) malloc(16000); 165252a38012Sejakowatz if (reply == 0) 165352a38012Sejakowatz return B_ERROR; 165452a38012Sejakowatz 165552a38012Sejakowatz msg.addon = in_dormant.addon; 165652a38012Sejakowatz msg.flavor_id = in_dormant.flavor_id; 165752a38012Sejakowatz msg.reply_port = _PortPool->GetPort(); 165852a38012Sejakowatz rv = write_port(port, SERVER_GET_DORMANT_FLAVOR_INFO, &msg, sizeof(msg)); 165952a38012Sejakowatz if (rv != B_OK) { 166052a38012Sejakowatz free(reply); 166152a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 166252a38012Sejakowatz return rv; 166352a38012Sejakowatz } 166452a38012Sejakowatz rv = read_port(msg.reply_port, &code, reply, 16000); 166552a38012Sejakowatz _PortPool->PutPort(msg.reply_port); 166652a38012Sejakowatz 166752a38012Sejakowatz if (rv < B_OK) { 166852a38012Sejakowatz free(reply); 166952a38012Sejakowatz return rv; 167052a38012Sejakowatz } 167152a38012Sejakowatz 167252a38012Sejakowatz if (reply->result == B_OK) 167352a38012Sejakowatz rv = out_flavor->Unflatten(reply->dfi_type, &reply->dfi, reply->dfi_size); 167452a38012Sejakowatz else 167552a38012Sejakowatz rv = reply->result; 167652a38012Sejakowatz 167752a38012Sejakowatz free(reply); 167852a38012Sejakowatz return rv; 167952a38012Sejakowatz } 168052a38012Sejakowatz 168152a38012Sejakowatz 168252a38012Sejakowatz status_t 168352a38012Sejakowatz BMediaRoster::GetLatencyFor(const media_node & producer, 168452a38012Sejakowatz bigtime_t * out_latency) 168552a38012Sejakowatz { 168652a38012Sejakowatz UNIMPLEMENTED(); 168752a38012Sejakowatz *out_latency = 0; 168852a38012Sejakowatz return B_ERROR; 168952a38012Sejakowatz } 169052a38012Sejakowatz 169152a38012Sejakowatz 169252a38012Sejakowatz status_t 169352a38012Sejakowatz BMediaRoster::GetInitialLatencyFor(const media_node & producer, 169452a38012Sejakowatz bigtime_t * out_latency, 169552a38012Sejakowatz uint32 * out_flags) 169652a38012Sejakowatz { 169752a38012Sejakowatz UNIMPLEMENTED(); 169852a38012Sejakowatz *out_latency = 0; 169952a38012Sejakowatz *out_flags = 0; 170052a38012Sejakowatz return B_ERROR; 170152a38012Sejakowatz } 170252a38012Sejakowatz 170352a38012Sejakowatz 170452a38012Sejakowatz status_t 170552a38012Sejakowatz BMediaRoster::GetStartLatencyFor(const media_node & time_source, 170652a38012Sejakowatz bigtime_t * out_latency) 170752a38012Sejakowatz { 170852a38012Sejakowatz UNIMPLEMENTED(); 170952a38012Sejakowatz *out_latency = 0; 171052a38012Sejakowatz return B_ERROR; 171152a38012Sejakowatz } 171252a38012Sejakowatz 171352a38012Sejakowatz 171452a38012Sejakowatz status_t 171552a38012Sejakowatz BMediaRoster::GetFileFormatsFor(const media_node & file_interface, 171652a38012Sejakowatz media_file_format * out_formats, 171752a38012Sejakowatz int32 * io_num_infos) 171852a38012Sejakowatz { 171952a38012Sejakowatz UNIMPLEMENTED(); 172052a38012Sejakowatz return B_ERROR; 172152a38012Sejakowatz } 172252a38012Sejakowatz 172352a38012Sejakowatz 172452a38012Sejakowatz status_t 172552a38012Sejakowatz BMediaRoster::SetRefFor(const media_node & file_interface, 172652a38012Sejakowatz const entry_ref & file, 172752a38012Sejakowatz bool create_and_truncate, 172852a38012Sejakowatz bigtime_t * out_length) /* if create is false */ 172952a38012Sejakowatz { 173052a38012Sejakowatz UNIMPLEMENTED(); 173152a38012Sejakowatz return B_ERROR; 173252a38012Sejakowatz } 173352a38012Sejakowatz 173452a38012Sejakowatz 173552a38012Sejakowatz status_t 173652a38012Sejakowatz BMediaRoster::GetRefFor(const media_node & node, 173752a38012Sejakowatz entry_ref * out_file, 173852a38012Sejakowatz BMimeType * mime_type) 173952a38012Sejakowatz { 174052a38012Sejakowatz UNIMPLEMENTED(); 174152a38012Sejakowatz return B_ERROR; 174252a38012Sejakowatz } 174352a38012Sejakowatz 174452a38012Sejakowatz 174552a38012Sejakowatz status_t 174652a38012Sejakowatz BMediaRoster::SniffRefFor(const media_node & file_interface, 174752a38012Sejakowatz const entry_ref & file, 174852a38012Sejakowatz BMimeType * mime_type, 174952a38012Sejakowatz float * out_capability) 175052a38012Sejakowatz { 175152a38012Sejakowatz UNIMPLEMENTED(); 175252a38012Sejakowatz return B_ERROR; 175352a38012Sejakowatz } 175452a38012Sejakowatz 175552a38012Sejakowatz 175652a38012Sejakowatz /* This is the generic "here's a file, now can someone please play it" interface */ 175752a38012Sejakowatz status_t 175852a38012Sejakowatz BMediaRoster::SniffRef(const entry_ref & file, 175952a38012Sejakowatz uint64 require_node_kinds, /* if you need an EntityInterface or BufferConsumer or something */ 176052a38012Sejakowatz dormant_node_info * out_node, 176152a38012Sejakowatz BMimeType * mime_type) 176252a38012Sejakowatz { 176352a38012Sejakowatz UNIMPLEMENTED(); 176452a38012Sejakowatz return B_ERROR; 176552a38012Sejakowatz } 176652a38012Sejakowatz 176752a38012Sejakowatz 176852a38012Sejakowatz status_t 176952a38012Sejakowatz BMediaRoster::GetDormantNodeForType(const BMimeType & type, 177052a38012Sejakowatz uint64 require_node_kinds, 177152a38012Sejakowatz dormant_node_info * out_node) 177252a38012Sejakowatz { 177352a38012Sejakowatz UNIMPLEMENTED(); 177452a38012Sejakowatz return B_ERROR; 177552a38012Sejakowatz } 177652a38012Sejakowatz 177752a38012Sejakowatz 177852a38012Sejakowatz status_t 177952a38012Sejakowatz BMediaRoster::GetReadFileFormatsFor(const dormant_node_info & in_node, 178052a38012Sejakowatz media_file_format * out_read_formats, 178152a38012Sejakowatz int32 in_read_count, 178252a38012Sejakowatz int32 * out_read_count) 178352a38012Sejakowatz { 178452a38012Sejakowatz UNIMPLEMENTED(); 178552a38012Sejakowatz return B_ERROR; 178652a38012Sejakowatz } 178752a38012Sejakowatz 178852a38012Sejakowatz 178952a38012Sejakowatz status_t 179052a38012Sejakowatz BMediaRoster::GetWriteFileFormatsFor(const dormant_node_info & in_node, 179152a38012Sejakowatz media_file_format * out_write_formats, 179252a38012Sejakowatz int32 in_write_count, 179352a38012Sejakowatz int32 * out_write_count) 179452a38012Sejakowatz { 179552a38012Sejakowatz UNIMPLEMENTED(); 179652a38012Sejakowatz return B_ERROR; 179752a38012Sejakowatz } 179852a38012Sejakowatz 179952a38012Sejakowatz 180052a38012Sejakowatz status_t 180152a38012Sejakowatz BMediaRoster::GetFormatFor(const media_output & output, 180252a38012Sejakowatz media_format * io_format, 180352a38012Sejakowatz uint32 flags) 180452a38012Sejakowatz { 180552a38012Sejakowatz UNIMPLEMENTED(); 180652a38012Sejakowatz return B_ERROR; 180752a38012Sejakowatz } 180852a38012Sejakowatz 180952a38012Sejakowatz 181052a38012Sejakowatz status_t 181152a38012Sejakowatz BMediaRoster::GetFormatFor(const media_input & input, 181252a38012Sejakowatz media_format * io_format, 181352a38012Sejakowatz uint32 flags) 181452a38012Sejakowatz { 181552a38012Sejakowatz UNIMPLEMENTED(); 181652a38012Sejakowatz return B_ERROR; 181752a38012Sejakowatz } 181852a38012Sejakowatz 181952a38012Sejakowatz 182052a38012Sejakowatz status_t 182152a38012Sejakowatz BMediaRoster::GetFormatFor(const media_node & node, 182252a38012Sejakowatz media_format * io_format, 182352a38012Sejakowatz float quality) 182452a38012Sejakowatz { 182552a38012Sejakowatz UNIMPLEMENTED(); 182652a38012Sejakowatz return B_ERROR; 182752a38012Sejakowatz } 182852a38012Sejakowatz 182952a38012Sejakowatz 183052a38012Sejakowatz ssize_t 183152a38012Sejakowatz BMediaRoster::GetNodeAttributesFor(const media_node & node, 183252a38012Sejakowatz media_node_attribute * outArray, 183352a38012Sejakowatz size_t inMaxCount) 183452a38012Sejakowatz { 183552a38012Sejakowatz UNIMPLEMENTED(); 183652a38012Sejakowatz return B_ERROR; 183752a38012Sejakowatz } 183852a38012Sejakowatz 183952a38012Sejakowatz 184052a38012Sejakowatz media_node_id 184152a38012Sejakowatz BMediaRoster::NodeIDFor(port_id source_or_destination_port) 184252a38012Sejakowatz { 18439e9417d2Sbeveloper CALLED(); 18449e9417d2Sbeveloper 18459e9417d2Sbeveloper server_node_id_for_request request; 18469e9417d2Sbeveloper server_node_id_for_reply reply; 18479e9417d2Sbeveloper status_t rv; 18489e9417d2Sbeveloper 18499e9417d2Sbeveloper request.port = source_or_destination_port; 18509e9417d2Sbeveloper 18519e9417d2Sbeveloper rv = QueryServer(SERVER_NODE_ID_FOR, &request, sizeof(request), &reply, sizeof(reply)); 18529e9417d2Sbeveloper if (rv != B_OK) { 18539e9417d2Sbeveloper TRACE("BMediaRoster::NodeIDFor: failed (error %#lx)\n", rv); 18549e9417d2Sbeveloper return -1; 18559e9417d2Sbeveloper } 18569e9417d2Sbeveloper 18579e9417d2Sbeveloper return reply.nodeid; 185852a38012Sejakowatz } 185952a38012Sejakowatz 186052a38012Sejakowatz 186152a38012Sejakowatz status_t 186252a38012Sejakowatz BMediaRoster::GetInstancesFor(media_addon_id addon, 186352a38012Sejakowatz int32 flavor, 186452a38012Sejakowatz media_node_id * out_id, 186552a38012Sejakowatz int32 * io_count) 186652a38012Sejakowatz { 186785dfab2bSbeveloper CALLED(); 186885dfab2bSbeveloper if (out_id == NULL || io_count == NULL) 186985dfab2bSbeveloper return B_BAD_VALUE; 187085dfab2bSbeveloper if (*io_count <= 0) 187185dfab2bSbeveloper return B_BAD_VALUE; 187285dfab2bSbeveloper 187385dfab2bSbeveloper server_get_instances_for_request request; 187485dfab2bSbeveloper server_get_instances_for_reply reply; 187585dfab2bSbeveloper status_t rv; 187685dfab2bSbeveloper 187785dfab2bSbeveloper request.maxcount = *io_count; 187885dfab2bSbeveloper request.addon_id = addon; 187985dfab2bSbeveloper request.addon_flavor_id = flavor; 188085dfab2bSbeveloper 188185dfab2bSbeveloper rv = QueryServer(SERVER_GET_INSTANCES_FOR, &request, sizeof(request), &reply, sizeof(reply)); 188285dfab2bSbeveloper if (rv != B_OK) { 188385dfab2bSbeveloper TRACE("BMediaRoster::GetLiveNodes failed\n"); 188485dfab2bSbeveloper return rv; 188552a38012Sejakowatz } 188652a38012Sejakowatz 188785dfab2bSbeveloper *io_count = reply.count; 188885dfab2bSbeveloper if (reply.count > 0) 188985dfab2bSbeveloper memcpy(out_id, reply.node_id, sizeof(media_node_id) * reply.count); 189085dfab2bSbeveloper 189185dfab2bSbeveloper return B_OK; 189285dfab2bSbeveloper } 189352a38012Sejakowatz 189452a38012Sejakowatz 189552a38012Sejakowatz status_t 189652a38012Sejakowatz BMediaRoster::SetRealtimeFlags(uint32 in_enabled) 189752a38012Sejakowatz { 189852a38012Sejakowatz UNIMPLEMENTED(); 189952a38012Sejakowatz return B_ERROR; 190052a38012Sejakowatz } 190152a38012Sejakowatz 190252a38012Sejakowatz 190352a38012Sejakowatz status_t 190452a38012Sejakowatz BMediaRoster::GetRealtimeFlags(uint32 * out_enabled) 190552a38012Sejakowatz { 190652a38012Sejakowatz UNIMPLEMENTED(); 190752a38012Sejakowatz return B_ERROR; 190852a38012Sejakowatz } 190952a38012Sejakowatz 191052a38012Sejakowatz 191152a38012Sejakowatz ssize_t 191252a38012Sejakowatz BMediaRoster::AudioBufferSizeFor(int32 channel_count, 191352a38012Sejakowatz uint32 sample_format, 191452a38012Sejakowatz float frame_rate, 191552a38012Sejakowatz bus_type bus_kind) 191652a38012Sejakowatz { 191752a38012Sejakowatz UNIMPLEMENTED(); 191852a38012Sejakowatz return 4096; 191952a38012Sejakowatz } 192052a38012Sejakowatz 192152a38012Sejakowatz 192252a38012Sejakowatz /* Use MediaFlags to inquire about specific features of the Media Kit. */ 192352a38012Sejakowatz /* Returns < 0 for "not present", positive size for output data size. */ 192452a38012Sejakowatz /* 0 means that the capability is present, but no data about it. */ 192552a38012Sejakowatz /* static */ ssize_t 192652a38012Sejakowatz BMediaRoster::MediaFlags(media_flags cap, 192752a38012Sejakowatz void * buf, 192852a38012Sejakowatz size_t maxSize) 192952a38012Sejakowatz { 193052a38012Sejakowatz UNIMPLEMENTED(); 193152a38012Sejakowatz return 0; 193252a38012Sejakowatz } 193352a38012Sejakowatz 193452a38012Sejakowatz 193552a38012Sejakowatz 193652a38012Sejakowatz /* BLooper overrides */ 193752a38012Sejakowatz /* virtual */ void 193852a38012Sejakowatz BMediaRoster::MessageReceived(BMessage * message) 193952a38012Sejakowatz { 194052a38012Sejakowatz UNIMPLEMENTED(); 194152a38012Sejakowatz } 194252a38012Sejakowatz 194352a38012Sejakowatz /* virtual */ bool 194452a38012Sejakowatz BMediaRoster::QuitRequested() 194552a38012Sejakowatz { 194652a38012Sejakowatz UNIMPLEMENTED(); 194752a38012Sejakowatz return true; 194852a38012Sejakowatz } 194952a38012Sejakowatz 195052a38012Sejakowatz /* virtual */ BHandler * 195152a38012Sejakowatz BMediaRoster::ResolveSpecifier(BMessage *msg, 195252a38012Sejakowatz int32 index, 195352a38012Sejakowatz BMessage *specifier, 195452a38012Sejakowatz int32 form, 195552a38012Sejakowatz const char *property) 195652a38012Sejakowatz { 195752a38012Sejakowatz UNIMPLEMENTED(); 195852a38012Sejakowatz return 0; 195952a38012Sejakowatz } 196052a38012Sejakowatz 196152a38012Sejakowatz 196252a38012Sejakowatz /* virtual */ status_t 196352a38012Sejakowatz BMediaRoster::GetSupportedSuites(BMessage *data) 196452a38012Sejakowatz { 196552a38012Sejakowatz UNIMPLEMENTED(); 196652a38012Sejakowatz return B_ERROR; 196752a38012Sejakowatz } 196852a38012Sejakowatz 196952a38012Sejakowatz 197052a38012Sejakowatz BMediaRoster::~BMediaRoster() 197152a38012Sejakowatz { 197252a38012Sejakowatz CALLED(); 197352a38012Sejakowatz BMessage msg(MEDIA_SERVER_UNREGISTER_APP); 197452a38012Sejakowatz BMessage reply; 197552a38012Sejakowatz msg.AddInt32("team",team); 1976f4925104Sbeveloper QueryServer(&msg, &reply); 197752a38012Sejakowatz } 197852a38012Sejakowatz 197952a38012Sejakowatz 198052a38012Sejakowatz /************************************************************* 198152a38012Sejakowatz * private BMediaRoster 198252a38012Sejakowatz *************************************************************/ 198352a38012Sejakowatz 198452a38012Sejakowatz // deprecated call 198552a38012Sejakowatz status_t 198652a38012Sejakowatz BMediaRoster::SetOutputBuffersFor(const media_source & output, 198752a38012Sejakowatz BBufferGroup * group, 198852a38012Sejakowatz bool will_reclaim ) 198952a38012Sejakowatz { 199052a38012Sejakowatz UNIMPLEMENTED(); 19919e9417d2Sbeveloper debugger("BMediaRoster::SetOutputBuffersFor missing\n"); 199252a38012Sejakowatz return B_ERROR; 199352a38012Sejakowatz } 199452a38012Sejakowatz 199552a38012Sejakowatz 199652a38012Sejakowatz /* FBC stuffing (Mmmh, Stuffing!) */ 199752a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_0(void *) { return B_ERROR; } 199852a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_1(void *) { return B_ERROR; } 199952a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_2(void *) { return B_ERROR; } 200052a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_3(void *) { return B_ERROR; } 200152a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_4(void *) { return B_ERROR; } 200252a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_5(void *) { return B_ERROR; } 200352a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_6(void *) { return B_ERROR; } 200452a38012Sejakowatz status_t BMediaRoster::_Reserved_MediaRoster_7(void *) { return B_ERROR; } 200552a38012Sejakowatz 200652a38012Sejakowatz 200752a38012Sejakowatz BMediaRoster::BMediaRoster() : 200852a38012Sejakowatz BLooper("BMediaRoster looper",B_NORMAL_PRIORITY,B_LOOPER_PORT_DEFAULT_CAPACITY) 200952a38012Sejakowatz { 201052a38012Sejakowatz CALLED(); 201152a38012Sejakowatz BMessage msg(MEDIA_SERVER_REGISTER_APP); 201252a38012Sejakowatz BMessage reply; 201352a38012Sejakowatz msg.AddInt32("team",team); 2014f4925104Sbeveloper QueryServer(&msg,&reply); 201552a38012Sejakowatz } 201652a38012Sejakowatz 201752a38012Sejakowatz /* static */ status_t 201852a38012Sejakowatz BMediaRoster::ParseCommand(BMessage & reply) 201952a38012Sejakowatz { 202052a38012Sejakowatz UNIMPLEMENTED(); 202152a38012Sejakowatz return B_ERROR; 202252a38012Sejakowatz } 202352a38012Sejakowatz 202452a38012Sejakowatz 202552a38012Sejakowatz 202652a38012Sejakowatz status_t 202752a38012Sejakowatz BMediaRoster::GetDefaultInfo(media_node_id for_default, 202852a38012Sejakowatz BMessage & out_config) 202952a38012Sejakowatz { 203052a38012Sejakowatz UNIMPLEMENTED(); 203152a38012Sejakowatz return B_ERROR; 203252a38012Sejakowatz } 203352a38012Sejakowatz 203452a38012Sejakowatz 203552a38012Sejakowatz 203652a38012Sejakowatz status_t 203752a38012Sejakowatz BMediaRoster::SetRunningDefault(media_node_id for_default, 203852a38012Sejakowatz const media_node & node) 203952a38012Sejakowatz { 204052a38012Sejakowatz UNIMPLEMENTED(); 204152a38012Sejakowatz return B_ERROR; 204252a38012Sejakowatz } 204352a38012Sejakowatz 204452a38012Sejakowatz 204552a38012Sejakowatz /************************************************************* 204652a38012Sejakowatz * static BMediaRoster variables 204752a38012Sejakowatz *************************************************************/ 204852a38012Sejakowatz 204952a38012Sejakowatz bool BMediaRoster::_isMediaServer; 205052a38012Sejakowatz port_id BMediaRoster::_mReplyPort; 205152a38012Sejakowatz int32 BMediaRoster::_mReplyPortRes; 205252a38012Sejakowatz int32 BMediaRoster::_mReplyPortUnavailCount; 205352a38012Sejakowatz BMediaRoster * BMediaRoster::_sDefault = NULL; 205452a38012Sejakowatz 2055