mirror of
https://github.com/albfan/miraclecast.git
synced 2025-03-09 23:38:56 +00:00
In this revision, the procedure of session negotiation is not yet completed but only TCP connection can be established between peers. You can try it by run script `res/miracle-wfdctl-demo` to see how it work.
417 lines
9 KiB
C
417 lines
9 KiB
C
/*
|
|
* MiracleCast - Wifi-Display/Miracast Implementation
|
|
*
|
|
* Copyright (c) 2013-2014 David Herrmann <dh.herrmann@gmail.com>
|
|
*
|
|
* MiracleCast is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation; either version 2.1 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* MiracleCast is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with MiracleCast; If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <systemd/sd-bus.h>
|
|
#include "ctl.h"
|
|
#include "util.h"
|
|
#include "wfd-dbus.h"
|
|
|
|
struct wfd_dbus
|
|
{
|
|
sd_bus *bus;
|
|
sd_event *loop;
|
|
|
|
bool exposed : 1;
|
|
};
|
|
|
|
int wfd_dbus_new(struct wfd_dbus **out, sd_event *loop, sd_bus *bus)
|
|
{
|
|
struct wfd_dbus *wfd_dbus = calloc(1, sizeof(struct wfd_dbus));
|
|
if(!wfd_dbus) {
|
|
return -ENOMEM;
|
|
}
|
|
|
|
wfd_dbus->bus = sd_bus_ref(bus);
|
|
wfd_dbus->loop = sd_event_ref(loop);
|
|
|
|
*out = wfd_dbus;
|
|
return 0;
|
|
}
|
|
|
|
void wfd_dbus_free(struct wfd_dbus *wfd_dbus)
|
|
{
|
|
if(!wfd_dbus) {
|
|
return;
|
|
}
|
|
|
|
if(wfd_dbus->exposed) {
|
|
sd_bus_release_name(wfd_dbus->bus, "org.freedesktop.miracle.wfd");
|
|
}
|
|
|
|
if(wfd_dbus->bus) {
|
|
sd_bus_unref(wfd_dbus->bus);
|
|
}
|
|
|
|
if(wfd_dbus->loop) {
|
|
sd_event_unref(wfd_dbus->loop);
|
|
}
|
|
|
|
free(wfd_dbus);
|
|
}
|
|
|
|
static inline int wfd_dbus_get_sink_path(struct wfd_sink *s, char **out)
|
|
{
|
|
return sd_bus_path_encode("/org/freedesktop/miracle/wfd/sink",
|
|
wfd_sink_get_label(s),
|
|
out);
|
|
}
|
|
|
|
static inline int wfd_dbus_get_session_path(struct wfd_session *s, char **out)
|
|
{
|
|
char buf[64];
|
|
int r = snprintf(buf, sizeof(buf), "%" PRIu64, s->id);
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
return sd_bus_path_encode("/org/freedesktop/miracle/wfd/session",
|
|
buf,
|
|
out);
|
|
}
|
|
|
|
static int wfd_dbus_enum(sd_bus *bus,
|
|
const char *path,
|
|
void *userdata,
|
|
char ***out,
|
|
sd_bus_error *out_error)
|
|
{
|
|
int r = 0, i = 0;
|
|
char **nodes, *node;
|
|
struct wfd_dbus *wfd_dbus = userdata;
|
|
struct ctl_wfd *wfd = ctl_wfd_get();
|
|
struct wfd_sink *sink;
|
|
|
|
if(strcmp("/org/freedesktop/miracle/wfd", path)) {
|
|
return 0;
|
|
}
|
|
|
|
if(!wfd->n_sinks) {
|
|
return 0;
|
|
}
|
|
|
|
nodes = malloc((wfd->n_sinks + 1) * sizeof(char *));
|
|
if(!nodes) {
|
|
return -ENOMEM;
|
|
}
|
|
|
|
ctl_wfd_foreach_sink(sink, wfd) {
|
|
r = wfd_dbus_get_sink_path(sink, &node);
|
|
if(0 > r) {
|
|
goto free_nodes;
|
|
}
|
|
nodes[i ++] = node;
|
|
}
|
|
|
|
nodes[i ++] = NULL;
|
|
*out = nodes;
|
|
|
|
return 0;
|
|
|
|
free_nodes:
|
|
while(i --) {
|
|
free(nodes[i]);
|
|
}
|
|
free(nodes);
|
|
return r;
|
|
}
|
|
|
|
int wfd_dbus_notify_new_sink(struct wfd_dbus *wfd_dbus, const char *p2p_mac)
|
|
{
|
|
_shl_free_ char *path;
|
|
_sd_bus_message_unref_ sd_bus_message *m;
|
|
int r = sd_bus_message_new_signal(wfd_dbus->bus,
|
|
&m,
|
|
"/org/freedesktop/miracle/wfd",
|
|
"org.freedesktop.DBus.ObjectManager",
|
|
"InterfaceAdded");
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
r = sd_bus_path_encode("/org/freedesktop/miracle/wfd/sink", p2p_mac, &path);
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
r = sd_bus_message_append(m, "o", path);
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
r = sd_bus_message_open_container(m, 'a', "{sa{sv}}");
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
r = sd_bus_message_append(m, "{sa{sv}}", "org.freedesktop.miracle.wfd.Sink", 0);
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
r = sd_bus_message_open_container(m, 'e', "sa{sv}");
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
/*r = sd_bus_message_append(m, "{sa{sv}}", "org.freedesktop.DBus.Peer", 0);*/
|
|
/*if (r < 0)*/
|
|
/*return r;*/
|
|
/*r = sd_bus_message_append(m, "{sa{sv}}", "org.freedesktop.DBus.Introspectable", 0);*/
|
|
/*if (r < 0)*/
|
|
/*return r;*/
|
|
/*r = sd_bus_message_append(m, "{sa{sv}}", "org.freedesktop.DBus.Properties", 0);*/
|
|
/*if (r < 0)*/
|
|
/*return r;*/
|
|
/*r = sd_bus_message_append(m, "{sa{sv}}", "org.freedesktop.DBus.ObjectManager", 0);*/
|
|
/*if (r < 0)*/
|
|
/*return r;*/
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_find_sink(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
void *userdata,
|
|
void **ret_found,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
_shl_free_ char *node = NULL;
|
|
struct wfd_dbus *wfd_dbus = userdata;
|
|
struct wfd_sink *sink;
|
|
int r = sd_bus_path_decode(path,
|
|
"/org/freedesktop/miracle/wfd/sink",
|
|
&node);
|
|
if(0 >= r || !node) {
|
|
return r;
|
|
}
|
|
|
|
r = ctl_wfd_find_sink_by_label(ctl_wfd_get(), node, &sink);
|
|
if(r) {
|
|
*ret_found = sink;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static int wfd_dbus_find_session(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
void *userdata,
|
|
void **ret_found,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_sink_start_session(sd_bus_message *m,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
struct wfd_sink *sink = userdata;
|
|
struct wfd_session *session;
|
|
_shl_free_ char *path = NULL;
|
|
int r = wfd_sink_start_session(sink, &session);
|
|
if(0 > r) {
|
|
return r;
|
|
}
|
|
|
|
r = wfd_dbus_get_session_path(session, &path);
|
|
if(0 > r) {
|
|
goto free_session;
|
|
}
|
|
|
|
log_debug("path=%s", path);
|
|
r = sd_bus_reply_method_return(m, "o", path);
|
|
if(0 <= r) {
|
|
return r;
|
|
}
|
|
|
|
free_session:
|
|
wfd_session_free(session);
|
|
return r;
|
|
}
|
|
|
|
static int wfd_dbus_sink_get_audio_formats(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_sink_get_video_formats(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_sink_has_video(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_sink_get_peer(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_sink_has_audio(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_session_end(sd_bus_message *m,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_session_get_sink(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int wfd_dbus_session_get_state(sd_bus *bus,
|
|
const char *path,
|
|
const char *interface,
|
|
const char *property,
|
|
sd_bus_message *reply,
|
|
void *userdata,
|
|
sd_bus_error *ret_error)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static const sd_bus_vtable wfd_dbus_vtable[] = {
|
|
SD_BUS_VTABLE_START(0),
|
|
SD_BUS_VTABLE_END,
|
|
};
|
|
|
|
static const sd_bus_vtable wfd_dbus_sink_vtable[] = {
|
|
SD_BUS_VTABLE_START(0),
|
|
SD_BUS_METHOD("StartSession", NULL, "o", wfd_dbus_sink_start_session, 0),
|
|
/*SD_BUS_PROPERTY("AudioFormats", "a{sv}", wfd_dbus_sink_get_audio_formats, 0, SD_BUS_VTABLE_PROPERTY_CONST),*/
|
|
/*SD_BUS_PROPERTY("VideoFormats", "a{sv}", wfd_dbus_sink_get_video_formats, 0, SD_BUS_VTABLE_PROPERTY_CONST),*/
|
|
/*SD_BUS_PROPERTY("HasAudio", "b", wfd_dbus_sink_has_audio, 0, SD_BUS_VTABLE_PROPERTY_CONST),*/
|
|
/*SD_BUS_PROPERTY("HasVideo", "b", wfd_dbus_sink_has_video, 0, SD_BUS_VTABLE_PROPERTY_CONST),*/
|
|
/*SD_BUS_PROPERTY("Peer", "o", wfd_dbus_sink_get_peer, 0, SD_BUS_VTABLE_PROPERTY_CONST),*/
|
|
SD_BUS_VTABLE_END,
|
|
};
|
|
|
|
static const sd_bus_vtable wfd_dbus_session_vtable[] = {
|
|
SD_BUS_VTABLE_START(0),
|
|
SD_BUS_METHOD("End", NULL, NULL, wfd_dbus_session_end, 0),
|
|
SD_BUS_PROPERTY("Sink", "o", wfd_dbus_session_get_sink, 0, SD_BUS_VTABLE_PROPERTY_CONST),
|
|
SD_BUS_PROPERTY("State", "i", wfd_dbus_session_get_state, 0, SD_BUS_VTABLE_PROPERTY_EMITS_CHANGE),
|
|
SD_BUS_VTABLE_END,
|
|
};
|
|
|
|
int wfd_dbus_expose(struct wfd_dbus *wfd_dbus)
|
|
{
|
|
int r = sd_bus_add_object_vtable(wfd_dbus->bus,
|
|
NULL,
|
|
"/org/freedesktop/miracle/wfd",
|
|
"org.freedesktop.miracle.wfd.Display",
|
|
wfd_dbus_vtable,
|
|
wfd_dbus);
|
|
if(0 > r) {
|
|
goto end;
|
|
}
|
|
|
|
r = sd_bus_add_fallback_vtable(wfd_dbus->bus,
|
|
NULL,
|
|
"/org/freedesktop/miracle/wfd/sink",
|
|
"org.freedesktop.miracle.wfd.Sink",
|
|
wfd_dbus_sink_vtable,
|
|
wfd_dbus_find_sink,
|
|
wfd_dbus);
|
|
if(0 > r) {
|
|
goto end;
|
|
}
|
|
|
|
/*r = sd_bus_add_fallback_vtable(wfd_dbus->bus,*/
|
|
/*NULL,*/
|
|
/*"/org/freedesktop/miracle/wfd/session",*/
|
|
/*"org.freedesktop.miracle.wfd.Session",*/
|
|
/*wfd_dbus_session_vtable,*/
|
|
/*wfd_dbus_find_session,*/
|
|
/*wfd_dbus);*/
|
|
/*if(0 > r) {*/
|
|
/*goto end;*/
|
|
/*}*/
|
|
|
|
r = sd_bus_add_node_enumerator(wfd_dbus->bus,
|
|
NULL,
|
|
"/org/freedesktop/miracle/wfd",
|
|
wfd_dbus_enum,
|
|
wfd_dbus);
|
|
if(0 > r) {
|
|
goto end;
|
|
}
|
|
|
|
r = sd_bus_add_object_manager(wfd_dbus->bus, NULL, "/org/freedesktop/miracle/wfd");
|
|
if(0 > r) {
|
|
goto end;
|
|
}
|
|
|
|
r = sd_bus_request_name(wfd_dbus->bus, "org.freedesktop.miracle.wfd", 0);
|
|
if(0 < r) {
|
|
wfd_dbus->exposed = true;
|
|
}
|
|
|
|
end:
|
|
return r;
|
|
}
|
|
|