owntone-server/src/outputs/airplay.c

4430 lines
121 KiB
C
Raw Normal View History

/*
* ALAC encoding adapted from airplay_play
* Copyright (C) 2005 Shiro Ninomiya <shiron@snino.com>
* GPLv2+
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <stdio.h>
#include <stdbool.h>
#include <unistd.h>
#include <stdint.h>
#include <inttypes.h>
#include <math.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <fcntl.h>
#include <time.h>
#include <arpa/inet.h>
#include <net/if.h>
#include <netinet/in.h>
#include <event2/event.h>
#include <event2/buffer.h>
#include <gcrypt.h>
#include "plist_wrap.h"
#include "evrtsp/evrtsp.h"
#include "conffile.h"
#include "logger.h"
#include "mdns.h"
#include "misc.h"
#include "player.h"
#include "db.h"
#include "artwork.h"
#include "dmap_common.h"
#include "rtp_common.h"
#include "outputs.h"
#include "pair.h"
// Airplay 2 has a gazallion parameters, many of them unknown to us. With the
// below it is possible to easily try different variations.
#define AIRPLAY_USE_STREAMID 0
#define AIRPLAY_USE_PAIRING_TRANSIENT 1
#define ALAC_HEADER_LEN 3
#define RAOP_QUALITY_SAMPLE_RATE_DEFAULT 44100
#define RAOP_QUALITY_BITS_PER_SAMPLE_DEFAULT 16
#define RAOP_QUALITY_CHANNELS_DEFAULT 2
// AirTunes v2 number of samples per packet
// Probably using this value because 44100/352 and 48000/352 has good 32 byte
// alignment, which improves performance of some encoders
#define RAOP_SAMPLES_PER_PACKET 352
#define RAOP_RTP_PAYLOADTYPE 0x60
// How many RTP packets keep in a buffer for retransmission
#define RAOP_PACKET_BUFFER_SIZE 1000
#define RAOP_MD_DELAY_STARTUP 15360
#define RAOP_MD_DELAY_SWITCH (RAOP_MD_DELAY_STARTUP * 2)
#define RAOP_MD_WANTS_TEXT (1 << 0)
#define RAOP_MD_WANTS_ARTWORK (1 << 1)
#define RAOP_MD_WANTS_PROGRESS (1 << 2)
// ATV4 and Homepod disconnect for reasons that are not clear, but sending them
// progress metadata at regular intervals reduces the problem. The below
// interval was determined via testing, see:
// https://github.com/ejurgensen/forked-daapd/issues/734#issuecomment-622959334
#define RAOP_KEEP_ALIVE_INTERVAL 25
// This is an arbitrary value which just needs to be kept in sync with the config
#define RAOP_CONFIG_MAX_VOLUME 11
const char *pair_device_id = "AABBCCDD11223344"; // TODO use actual ID
union sockaddr_all
{
struct sockaddr_in sin;
struct sockaddr_in6 sin6;
struct sockaddr sa;
struct sockaddr_storage ss;
};
enum airplay_devtype {
RAOP_DEV_APEX1_80211G,
RAOP_DEV_APEX2_80211N,
RAOP_DEV_APEX3_80211N,
RAOP_DEV_APPLETV,
RAOP_DEV_APPLETV4,
RAOP_DEV_HOMEPOD,
RAOP_DEV_OTHER,
};
// Session is starting up
#define AIRPLAY_STATE_F_STARTUP (1 << 13)
// Streaming is up (connection established)
#define AIRPLAY_STATE_F_CONNECTED (1 << 14)
// Couldn't start device
#define AIRPLAY_STATE_F_FAILED (1 << 15)
enum airplay_state {
// Device is stopped (no session)
AIRPLAY_STATE_STOPPED = 0,
// Session startup
AIRPLAY_STATE_STARTUP = AIRPLAY_STATE_F_STARTUP | 0x01,
AIRPLAY_STATE_OPTIONS = AIRPLAY_STATE_F_STARTUP | 0x02,
AIRPLAY_STATE_ANNOUNCE = AIRPLAY_STATE_F_STARTUP | 0x03,
AIRPLAY_STATE_SETUP = AIRPLAY_STATE_F_STARTUP | 0x04,
AIRPLAY_STATE_RECORD = AIRPLAY_STATE_F_STARTUP | 0x05,
// Session established
// - streaming ready (RECORD sent and acked, connection established)
// - commands (SET_PARAMETER) are possible
AIRPLAY_STATE_CONNECTED = AIRPLAY_STATE_F_CONNECTED | 0x01,
// Media data is being sent
AIRPLAY_STATE_STREAMING = AIRPLAY_STATE_F_CONNECTED | 0x02,
// Session teardown in progress (-> going to STOPPED state)
AIRPLAY_STATE_TEARDOWN = AIRPLAY_STATE_F_CONNECTED | 0x03,
// Session is failed, couldn't startup or error occurred
AIRPLAY_STATE_FAILED = AIRPLAY_STATE_F_FAILED | 0x01,
// Password issue: unknown password or bad password, or pending PIN from user
AIRPLAY_STATE_PASSWORD = AIRPLAY_STATE_F_FAILED | 0x02,
};
enum airplay_seq_type
{
AIRPLAY_SEQ_ABORT = -1,
AIRPLAY_SEQ_START,
AIRPLAY_SEQ_START_RERUN,
AIRPLAY_SEQ_START_AP2,
AIRPLAY_SEQ_PROBE,
AIRPLAY_SEQ_FLUSH,
AIRPLAY_SEQ_STOP,
AIRPLAY_SEQ_FAILURE,
AIRPLAY_SEQ_PIN_START,
AIRPLAY_SEQ_SEND_VOLUME,
AIRPLAY_SEQ_SEND_TEXT,
AIRPLAY_SEQ_SEND_PROGRESS,
AIRPLAY_SEQ_SEND_ARTWORK,
AIRPLAY_SEQ_PAIR_SETUP,
AIRPLAY_SEQ_PAIR_VERIFY,
AIRPLAY_SEQ_PAIR_TRANSIENT,
AIRPLAY_SEQ_FEEDBACK,
AIRPLAY_SEQ_CONTINUE, // Must be last element
};
// Info about the device, which is not required by the player, only internally
struct airplay_extra
{
enum airplay_devtype devtype;
uint16_t wanted_metadata;
bool encrypt;
bool supports_auth_setup;
bool supports_pairing_transient;
};
struct airplay_master_session
{
struct evbuffer *evbuf;
int evbuf_samples;
struct rtp_session *rtp_session;
struct rtcp_timestamp cur_stamp;
uint8_t *rawbuf;
size_t rawbuf_size;
int samples_per_packet;
bool encrypt;
// Number of samples that we tell the output to buffer (this will mean that
// the position that we send in the sync packages are offset by this amount
// compared to the rtptimes of the corresponding RTP packages we are sending)
int output_buffer_samples;
struct airplay_master_session *next;
};
struct airplay_session
{
uint64_t device_id;
int callback_id;
struct airplay_master_session *master_session;
struct evrtsp_connection *ctrl;
struct evrtsp_connection *event;
enum airplay_state state;
enum airplay_seq_type next_seq;
uint16_t wanted_metadata;
bool req_has_auth;
bool encrypt;
bool auth_quirk_itunes;
bool supports_post;
bool supports_auth_setup;
struct event *deferredev;
int reqs_in_flight;
int cseq;
char *session;
uint32_t session_id;
char session_url[128];
char *realm;
char *nonce;
const char *password;
char *devname;
char *address;
int family;
int volume;
char *local_address;
unsigned short data_port;
unsigned short control_port;
unsigned short events_port;
unsigned short timing_port; // ATV4 has this set to 0, but it is not used by forked-daapd anyway
/* Pairing, see pair.h */
enum pair_type pair_type;
struct pair_cipher_context *control_cipher_ctx;
struct pair_cipher_context *events_cipher_ctx;
struct pair_verify_context *pair_verify_ctx;
struct pair_setup_context *pair_setup_ctx;
uint8_t shared_secret[32];
int server_fd;
int events_fd;
union sockaddr_all sa;
struct airplay_service *timing_svc;
struct airplay_service *control_svc;
struct airplay_session *next;
};
struct airplay_metadata
{
struct evbuffer *metadata;
struct evbuffer *artwork;
int artwork_fmt;
};
struct airplay_service
{
int fd;
unsigned short port;
struct event *ev;
};
/* NTP timestamp definitions */
#define FRAC 4294967296. /* 2^32 as a double */
#define NTP_EPOCH_DELTA 0x83aa7e80 /* 2208988800 - that's 1970 - 1900 in seconds */
// TODO move to rtp_common
struct ntp_stamp
{
uint32_t sec;
uint32_t frac;
};
/* --------------------------- SEQUENCE DEFINITIONS ------------------------- */
struct airplay_seq_definition
{
enum airplay_seq_type seq_type;
// Called when a sequence ends, successfully or not. Shoulds also, if
// required, take care of notifying player and free the session.
void (*on_success)(struct airplay_session *rs);
void (*on_error)(struct airplay_session *rs);
};
struct airplay_seq_request
{
enum airplay_seq_type seq_type;
const char *name; // Name of request (for logging)
enum evrtsp_cmd_type rtsp_type;
int (*payload_make)(struct evrtsp_request *req, struct airplay_session *rs, void *arg);
enum airplay_seq_type (*response_handler)(struct evrtsp_request *req, struct airplay_session *rs);
const char *content_type;
const char *uri;
bool proceed_on_rtsp_not_ok; // If true return code != RTSP_OK will not abort the sequence
};
struct airplay_seq_ctx
{
struct airplay_seq_request *cur_request;
void (*on_success)(struct airplay_session *rs);
void (*on_error)(struct airplay_session *rs);
struct airplay_session *session;
void *payload_make_arg;
const char *log_caller;
};
/* ------------------------------ MISC GLOBALS ------------------------------ */
/*
static const uint8_t airplay_auth_setup_pubkey[] =
"\x59\x02\xed\xe9\x0d\x4e\xf2\xbd\x4c\xb6\x8a\x63\x30\x03\x82\x07"
"\xa9\x4d\xbd\x50\xd8\xaa\x46\x5b\x5d\x8c\x01\x2a\x0c\x7e\x1d\x4e";
*/
struct features_type_map
{
uint32_t bit;
char *name;
};
// List of features announced by AirPlay 2 speakers
// Credit @invano, see https://emanuelecozzi.net/docs/airplay2
static const struct features_type_map features_map[] =
{
{ 0, "SupportsAirPlayVideoV1" },
{ 1, "SupportsAirPlayPhoto" },
{ 5, "SupportsAirPlaySlideshow" },
{ 7, "SupportsAirPlayScreen" },
{ 9, "SupportsAirPlayAudio" },
{ 11, "AudioRedunant" },
{ 14, "Authentication_4" }, // FairPlay authentication
{ 15, "MetadataFeatures_0" }, // Send artwork image to receiver
{ 16, "MetadataFeatures_1" }, // Send track progress status to receiver
{ 17, "MetadataFeatures_2" }, // Send NowPlaying info via DAAP
{ 18, "AudioFormats_0" },
{ 19, "AudioFormats_1" },
{ 20, "AudioFormats_2" },
{ 21, "AudioFormats_3" },
{ 23, "Authentication_1" }, // RSA authentication (NA)
{ 26, "Authentication_8" }, // 26 || 51, MFi authentication
{ 27, "SupportsLegacyPairing" },
{ 30, "HasUnifiedAdvertiserInfo" },
{ 32, "IsCarPlay" },
{ 32, "SupportsVolume" }, // !32
{ 33, "SupportsAirPlayVideoPlayQueue" },
{ 34, "SupportsAirPlayFromCloud" }, // 34 && flags_6_SupportsAirPlayFromCloud
{ 35, "SupportsTLS_PSK" },
{ 38, "SupportsUnifiedMediaControl" },
{ 40, "SupportsBufferedAudio" }, // srcvers >= 354.54.6 && 40
{ 41, "SupportsPTP" }, // srcvers >= 366 && 41
{ 42, "SupportsScreenMultiCodec" },
{ 43, "SupportsSystemPairing" },
{ 44, "IsAPValeriaScreenSender" },
{ 46, "SupportsHKPairingAndAccessControl" },
{ 48, "SupportsCoreUtilsPairingAndEncryption" }, // 38 || 46 || 43 || 48
{ 49, "SupportsAirPlayVideoV2" },
{ 50, "MetadataFeatures_3" }, // Send NowPlaying info via bplist
{ 51, "SupportsUnifiedPairSetupAndMFi" },
{ 52, "SupportsSetPeersExtendedMessage" },
{ 54, "SupportsAPSync" },
{ 55, "SupportsWoL" }, // 55 || 56
{ 56, "SupportsWoL" }, // 55 || 56
{ 58, "SupportsHangdogRemoteControl" }, // ((isAppleTV || isAppleAudioAccessory) && 58) || (isThirdPartyTV && flags_10)
{ 59, "SupportsAudioStreamConnectionSetup" }, // 59 && !disableStreamConnectionSetup
{ 60, "SupportsAudioMediaDataControl" }, // 59 && 60 && !disableMediaDataControl
{ 61, "SupportsRFC2198Redundancy" },
};
/* Keep in sync with enum airplay_devtype */
static const char *airplay_devtype[] =
{
"AirPort Express 1 - 802.11g",
"AirPort Express 2 - 802.11n",
"AirPort Express 3 - 802.11n",
"AppleTV",
"AppleTV4",
"HomePod",
"Other",
};
/* Struct with default quality levels */
static struct media_quality airplay_quality_default =
{
RAOP_QUALITY_SAMPLE_RATE_DEFAULT,
RAOP_QUALITY_BITS_PER_SAMPLE_DEFAULT,
RAOP_QUALITY_CHANNELS_DEFAULT
};
/* From player.c */
extern struct event_base *evbase_player;
/* AirTunes v2 time synchronization */
static struct airplay_service timing_4svc;
static struct airplay_service timing_6svc;
/* AirTunes v2 playback synchronization / control */
static struct airplay_service control_4svc;
static struct airplay_service control_6svc;
/* Metadata */
static struct output_metadata *airplay_cur_metadata;
/* Keep-alive timer - hack for ATV's with tvOS 10 */
static struct event *keep_alive_timer;
static struct timeval keep_alive_tv = { RAOP_KEEP_ALIVE_INTERVAL, 0 };
/* Sessions */
static struct airplay_master_session *airplay_master_sessions;
static struct airplay_session *airplay_sessions;
// Forwards
static int
airplay_device_start(struct output_device *rd, int callback_id);
static void
sequence_start(enum airplay_seq_type seq_type, struct airplay_session *rs, void *arg, const char *log_caller);
static void
sequence_continue(struct airplay_seq_ctx *seq_ctx);
/* ------------------------------- MISC HELPERS ----------------------------- */
/* ALAC bits writer - big endian
* p outgoing buffer pointer
* val bitfield value
* blen bitfield length, max 8 bits
* bpos bit position in the current byte (pointed by *p)
*/
static inline void
alac_write_bits(uint8_t **p, uint8_t val, int blen, int *bpos)
{
int lb;
int rb;
int bd;
/* Remaining bits in the current byte */
lb = 7 - *bpos + 1;
/* Number of bits overflowing */
rb = lb - blen;
if (rb >= 0)
{
bd = val << rb;
if (*bpos == 0)
**p = bd;
else
**p |= bd;
/* No over- nor underflow, we're done with this byte */
if (rb == 0)
{
*p += 1;
*bpos = 0;
}
else
*bpos += blen;
}
else
{
/* Fill current byte */
bd = val >> -rb;
**p |= bd;
/* Overflow goes to the next byte */
*p += 1;
**p = val << (8 + rb);
*bpos = -rb;
}
}
/* Raw data must be little endian */
static void
alac_encode(uint8_t *dst, uint8_t *raw, int len)
{
uint8_t *maxraw;
int bpos;
bpos = 0;
maxraw = raw + len;
alac_write_bits(&dst, 1, 3, &bpos); /* channel=1, stereo */
alac_write_bits(&dst, 0, 4, &bpos); /* unknown */
alac_write_bits(&dst, 0, 8, &bpos); /* unknown */
alac_write_bits(&dst, 0, 4, &bpos); /* unknown */
alac_write_bits(&dst, 0, 1, &bpos); /* hassize */
alac_write_bits(&dst, 0, 2, &bpos); /* unused */
alac_write_bits(&dst, 1, 1, &bpos); /* is-not-compressed */
for (; raw < maxraw; raw += 4)
{
/* Byteswap to big endian */
alac_write_bits(&dst, *(raw + 1), 8, &bpos);
alac_write_bits(&dst, *raw, 8, &bpos);
alac_write_bits(&dst, *(raw + 3), 8, &bpos);
alac_write_bits(&dst, *(raw + 2), 8, &bpos);
}
}
/* AirTunes v2 time synchronization helpers */
static inline void
timespec_to_ntp(struct timespec *ts, struct ntp_stamp *ns)
{
/* Seconds since NTP Epoch (1900-01-01) */
ns->sec = ts->tv_sec + NTP_EPOCH_DELTA;
ns->frac = (uint32_t)((double)ts->tv_nsec * 1e-9 * FRAC);
}
static inline void
ntp_to_timespec(struct ntp_stamp *ns, struct timespec *ts)
{
/* Seconds since Unix Epoch (1970-01-01) */
ts->tv_sec = ns->sec - NTP_EPOCH_DELTA;
ts->tv_nsec = (long)((double)ns->frac / (1e-9 * FRAC));
}
static inline int
airplay_timing_get_clock_ntp(struct ntp_stamp *ns)
{
struct timespec ts;
int ret;
ret = clock_gettime(CLOCK_MONOTONIC, &ts);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't get clock: %s\n", strerror(errno));
return -1;
}
timespec_to_ntp(&ts, ns);
return 0;
}
/* ----------------------- RAOP crypto stuff - from VLC --------------------- */
static int
encrypt_chacha(uint8_t *cipher, uint8_t *plain, size_t plain_len, const uint8_t *key, size_t key_len, const void *ad, size_t ad_len, uint8_t *tag, size_t tag_len, uint8_t *nonce, size_t nonce_len)
{
gcry_cipher_hd_t hd;
if (gcry_cipher_open(&hd, GCRY_CIPHER_CHACHA20, GCRY_CIPHER_MODE_POLY1305, 0) != GPG_ERR_NO_ERROR)
return -1;
if (gcry_cipher_setkey(hd, key, key_len) != GPG_ERR_NO_ERROR)
goto error;
if (gcry_cipher_setiv(hd, nonce, nonce_len) != GPG_ERR_NO_ERROR)
goto error;
if (gcry_cipher_authenticate(hd, ad, ad_len) != GPG_ERR_NO_ERROR)
goto error;
if (gcry_cipher_encrypt(hd, cipher, plain_len, plain, plain_len) != GPG_ERR_NO_ERROR)
goto error;
if (gcry_cipher_gettag(hd, tag, tag_len) != GPG_ERR_NO_ERROR)
goto error;
gcry_cipher_close(hd);
return 0;
error:
gcry_cipher_close(hd);
return -1;
}
/* ------------------ Helpers for sending RAOP/RTSP requests ---------------- */
static int
request_header_auth_add(struct evrtsp_request *req, struct airplay_session *rs, const char *method, const char *uri)
{
char ha1[33];
char ha2[33];
char ebuf[64];
char auth[256];
const char *hash_fmt;
const char *username;
uint8_t *hash_bytes;
size_t hashlen;
gcry_md_hd_t hd;
gpg_error_t gc_err;
int i;
int ret;
rs->req_has_auth = 0;
if (!rs->nonce)
return 0;
if (!rs->password)
{
DPRINTF(E_LOG, L_RAOP, "Authentication required but no password found for device '%s'\n", rs->devname);
return -2;
}
if (rs->auth_quirk_itunes)
{
hash_fmt = "%02X"; /* Uppercase hex */
username = "iTunes";
}
else
{
hash_fmt = "%02x";
username = ""; /* No username */
}
gc_err = gcry_md_open(&hd, GCRY_MD_MD5, 0);
if (gc_err != GPG_ERR_NO_ERROR)
{
gpg_strerror_r(gc_err, ebuf, sizeof(ebuf));
DPRINTF(E_LOG, L_RAOP, "Could not open MD5: %s\n", ebuf);
return -1;
}
memset(ha1, 0, sizeof(ha1));
memset(ha2, 0, sizeof(ha2));
hashlen = gcry_md_get_algo_dlen(GCRY_MD_MD5);
/* HA 1 */
gcry_md_write(hd, username, strlen(username));
gcry_md_write(hd, ":", 1);
gcry_md_write(hd, rs->realm, strlen(rs->realm));
gcry_md_write(hd, ":", 1);
gcry_md_write(hd, rs->password, strlen(rs->password));
hash_bytes = gcry_md_read(hd, GCRY_MD_MD5);
if (!hash_bytes)
{
DPRINTF(E_LOG, L_RAOP, "Could not read MD5 hash\n");
return -1;
}
for (i = 0; i < hashlen; i++)
sprintf(ha1 + (2 * i), hash_fmt, hash_bytes[i]);
/* RESET */
gcry_md_reset(hd);
/* HA 2 */
gcry_md_write(hd, method, strlen(method));
gcry_md_write(hd, ":", 1);
gcry_md_write(hd, uri, strlen(uri));
hash_bytes = gcry_md_read(hd, GCRY_MD_MD5);
if (!hash_bytes)
{
DPRINTF(E_LOG, L_RAOP, "Could not read MD5 hash\n");
return -1;
}
for (i = 0; i < hashlen; i++)
sprintf(ha2 + (2 * i), hash_fmt, hash_bytes[i]);
/* RESET */
gcry_md_reset(hd);
/* Final value */
gcry_md_write(hd, ha1, 32);
gcry_md_write(hd, ":", 1);
gcry_md_write(hd, rs->nonce, strlen(rs->nonce));
gcry_md_write(hd, ":", 1);
gcry_md_write(hd, ha2, 32);
hash_bytes = gcry_md_read(hd, GCRY_MD_MD5);
if (!hash_bytes)
{
DPRINTF(E_LOG, L_RAOP, "Could not read MD5 hash\n");
return -1;
}
for (i = 0; i < hashlen; i++)
sprintf(ha1 + (2 * i), hash_fmt, hash_bytes[i]);
gcry_md_close(hd);
/* Build header */
ret = snprintf(auth, sizeof(auth), "Digest username=\"%s\", realm=\"%s\", nonce=\"%s\", uri=\"%s\", response=\"%s\"",
username, rs->realm, rs->nonce, uri, ha1);
if ((ret < 0) || (ret >= sizeof(auth)))
{
DPRINTF(E_LOG, L_RAOP, "Authorization value header exceeds buffer size\n");
return -1;
}
evrtsp_add_header(req->output_headers, "Authorization", auth);
DPRINTF(E_DBG, L_RAOP, "Authorization header: %s\n", auth);
rs->req_has_auth = 1;
return 0;
}
static int
response_header_auth_parse(struct airplay_session *rs, struct evrtsp_request *req)
{
const char *param;
char *auth;
char *token;
char *ptr;
if (rs->realm)
{
free(rs->realm);
rs->realm = NULL;
}
if (rs->nonce)
{
free(rs->nonce);
rs->nonce = NULL;
}
param = evrtsp_find_header(req->input_headers, "WWW-Authenticate");
if (!param)
{
DPRINTF(E_LOG, L_RAOP, "WWW-Authenticate header not found\n");
return -1;
}
DPRINTF(E_DBG, L_RAOP, "WWW-Authenticate: %s\n", param);
if (strncmp(param, "Digest ", strlen("Digest ")) != 0)
{
DPRINTF(E_LOG, L_RAOP, "Unsupported authentication method: %s\n", param);
return -1;
}
auth = strdup(param);
if (!auth)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for WWW-Authenticate header copy\n");
return -1;
}
token = strchr(auth, ' ');
token++;
token = strtok_r(token, " =", &ptr);
while (token)
{
if (strcmp(token, "realm") == 0)
{
token = strtok_r(NULL, "=\"", &ptr);
if (!token)
break;
rs->realm = strdup(token);
}
else if (strcmp(token, "nonce") == 0)
{
token = strtok_r(NULL, "=\"", &ptr);
if (!token)
break;
rs->nonce = strdup(token);
}
token = strtok_r(NULL, " =", &ptr);
}
free(auth);
if (!rs->realm || !rs->nonce)
{
DPRINTF(E_LOG, L_RAOP, "Could not find realm/nonce in WWW-Authenticate header\n");
if (rs->realm)
{
free(rs->realm);
rs->realm = NULL;
}
if (rs->nonce)
{
free(rs->nonce);
rs->nonce = NULL;
}
return -1;
}
DPRINTF(E_DBG, L_RAOP, "Found realm: [%s], nonce: [%s]\n", rs->realm, rs->nonce);
return 0;
}
static int
request_headers_add(struct evrtsp_request *req, struct airplay_session *rs, enum evrtsp_cmd_type req_method)
{
char buf[64];
const char *method;
const char *url;
const char *user_agent;
int ret;
method = evrtsp_method(req_method);
snprintf(buf, sizeof(buf), "%d", rs->cseq);
evrtsp_add_header(req->output_headers, "CSeq", buf);
rs->cseq++;
user_agent = cfg_getstr(cfg_getsec(cfg, "general"), "user_agent");
evrtsp_add_header(req->output_headers, "User-Agent", user_agent);
/* Add Authorization header */
url = (req_method == EVRTSP_REQ_OPTIONS) ? "*" : rs->session_url;
ret = request_header_auth_add(req, rs, method, url);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not add Authorization header\n");
if (ret == -2)
rs->state = AIRPLAY_STATE_PASSWORD;
return -1;
}
snprintf(buf, sizeof(buf), "%" PRIX64, libhash);
evrtsp_add_header(req->output_headers, "Client-Instance", buf);
evrtsp_add_header(req->output_headers, "DACP-ID", buf);
// We set Active-Remote as 32 bit unsigned decimal, as at least my device
// can't handle any larger. Must be aligned with volume_byactiveremote().
snprintf(buf, sizeof(buf), "%" PRIu32, (uint32_t)rs->device_id);
evrtsp_add_header(req->output_headers, "Active-Remote", buf);
if (rs->session)
evrtsp_add_header(req->output_headers, "Session", rs->session);
#if AIRPLAY_USE_STREAMID
evrtsp_add_header(req->output_headers, "X-Apple-StreamID", "1");
#endif
/* Content-Length added automatically by evrtsp */
return 0;
}
static int
session_url_set(struct airplay_session *rs)
{
char *address = NULL;
char *intf;
unsigned short port;
int family;
int ret;
// Determine local address, needed for SDP and session URL
evrtsp_connection_get_local_address(rs->ctrl, &address, &port, &family);
if (!address || (port == 0))
{
DPRINTF(E_LOG, L_RAOP, "Could not determine local address\n");
goto error;
}
intf = strchr(address, '%');
if (intf)
{
*intf = '\0';
intf++;
}
DPRINTF(E_DBG, L_RAOP, "Local address: %s (LL: %s) port %d\n", address, (intf) ? intf : "no", port);
// Session ID and session URL
gcry_randomize(&rs->session_id, sizeof(rs->session_id), GCRY_STRONG_RANDOM);
if (family == AF_INET)
ret = snprintf(rs->session_url, sizeof(rs->session_url), "rtsp://%s/%u", address, rs->session_id);
else
ret = snprintf(rs->session_url, sizeof(rs->session_url), "rtsp://[%s]/%u", address, rs->session_id);
if ((ret < 0) || (ret >= sizeof(rs->session_url)))
{
DPRINTF(E_LOG, L_RAOP, "Session URL length exceeds 127 characters\n");
goto error;
}
rs->local_address = address;
return 0;
error:
free(address);
return -1;
}
static void
metadata_rtptimes_get(uint32_t *start, uint32_t *display, uint32_t *pos, uint32_t *end, struct airplay_master_session *rms, struct output_metadata *metadata)
{
struct rtp_session *rtp_session = rms->rtp_session;
// All the calculations with long ints to avoid surprises
int64_t sample_rate;
int64_t diff_ms;
int64_t elapsed_ms;
int64_t elapsed_samples;
int64_t len_samples;
sample_rate = rtp_session->quality.sample_rate;
// First calculate the rtptime that streaming of this item started:
// - at time metadata->pts the elapsed time was metadata->pos_ms
// - the time is now rms->cur_stamp.ts and the position is rms->cur_stamp.pos
// -> time since item started is elapsed_ms = metadata->pos_ms + (rms->cur_stamp.ts - metadata->pts)
// -> start must then be start = rms->cur_stamp.pos - elapsed_ms * sample_rate;
diff_ms = (rms->cur_stamp.ts.tv_sec - metadata->pts.tv_sec) * 1000L + (rms->cur_stamp.ts.tv_nsec - metadata->pts.tv_nsec) / 1000000L;
elapsed_ms = (int64_t)metadata->pos_ms + diff_ms;
elapsed_samples = elapsed_ms * sample_rate / 1000;
*start = rms->cur_stamp.pos - elapsed_samples;
/* DPRINTF(E_DBG, L_RAOP, "pos_ms=%u, len_ms=%u, startup=%d, metadata.pts=%ld.%09ld, player.ts=%ld.%09ld, diff_ms=%" PRIi64 ", elapsed_ms=%" PRIi64 "\n",
metadata->pos_ms, metadata->len_ms, metadata->startup, metadata->pts.tv_sec, metadata->pts.tv_nsec, rms->cur_stamp.ts.tv_sec, rms->cur_stamp.ts.tv_nsec, diff_ms, elapsed_ms);
*/
// Here's the deal with progress values:
// - display is always start minus a delay
// -> delay x1 if streaming is starting for this device (joining or not)
// -> delay x2 if stream is switching to a new song
// TODO what if we are just sending a keep_alive?
// - pos is the RTP time of the first sample for this song for this device
// -> start of song
// -> start of song + offset if device is joining in the middle of a song,
// or getting out of a pause or seeking
// - end is the RTP time of the last sample for this song
len_samples = (int64_t)metadata->len_ms * sample_rate / 1000;
*display = metadata->startup ? *start - RAOP_MD_DELAY_STARTUP : *start - RAOP_MD_DELAY_SWITCH;
*pos = MAX(rms->cur_stamp.pos, *start);
*end = len_samples ? *start + len_samples : *pos;
DPRINTF(E_SPAM, L_RAOP, "start=%u, display=%u, pos=%u, end=%u, rtp_session.pos=%u, cur_stamp.pos=%u\n",
*start, *display, *pos, *end, rtp_session->pos, rms->cur_stamp.pos);
}
// TODO not clear if Airplay 2 uses this header
static int
rtpinfo_header_add(struct evrtsp_request *req, struct airplay_session *rs, struct output_metadata *metadata)
{
uint32_t start;
uint32_t display;
uint32_t pos;
uint32_t end;
char rtpinfo[32];
int ret;
metadata_rtptimes_get(&start, &display, &pos, &end, rs->master_session, metadata);
ret = snprintf(rtpinfo, sizeof(rtpinfo), "rtptime=%u", start);
if ((ret < 0) || (ret >= sizeof(rtpinfo)))
{
DPRINTF(E_LOG, L_RAOP, "RTP-Info too big for buffer while sending metadata\n");
return -1;
}
evrtsp_add_header(req->output_headers, "RTP-Info", rtpinfo);
return 0;
}
static void
rtsp_cipher(struct evbuffer *evbuf, void *arg, int encrypt)
{
struct airplay_session *rs = arg;
uint8_t *out = NULL;
size_t out_len = 0;
int ret;
uint8_t *in = evbuffer_pullup(evbuf, -1);
size_t in_len = evbuffer_get_length(evbuf);
if (encrypt)
{
if (in_len < 4096)
DHEXDUMP(E_DBG, L_RAOP, in, in_len, "Encrypting outgoing request\n");
else
DPRINTF(E_DBG, L_RAOP, "Encrypting outgoing request (size %zu)\n", in_len);
ret = pair_encrypt(&out, &out_len, in, in_len, rs->control_cipher_ctx);
}
else
{
ret = pair_decrypt(&out, &out_len, in, in_len, rs->control_cipher_ctx);
if (out_len < 4096)
DHEXDUMP(E_DBG, L_RAOP, out, out_len, "Decrypted incoming response\n");
else
DPRINTF(E_DBG, L_RAOP, "Decrypted incoming response (size %zu)\n", out_len);
}
evbuffer_drain(evbuf, in_len);
if (ret < 0)
{
// TODO test this error condition seems that it can lead to a freeze
DPRINTF(E_LOG, L_RAOP, "Error while ciphering: %s\n", pair_cipher_errmsg(rs->control_cipher_ctx));
return;
}
evbuffer_add(evbuf, out, out_len);
}
/* ------------------------------ Session handling -------------------------- */
// Maps our internal state to the generic output state and then makes a callback
// to the player to tell that state
static void
session_status(struct airplay_session *rs)
{
enum output_device_state state;
switch (rs->state)
{
case AIRPLAY_STATE_PASSWORD:
state = OUTPUT_STATE_PASSWORD;
break;
case AIRPLAY_STATE_FAILED:
state = OUTPUT_STATE_FAILED;
break;
case AIRPLAY_STATE_STOPPED:
state = OUTPUT_STATE_STOPPED;
break;
case AIRPLAY_STATE_STARTUP ... AIRPLAY_STATE_RECORD:
state = OUTPUT_STATE_STARTUP;
break;
case AIRPLAY_STATE_CONNECTED:
state = OUTPUT_STATE_CONNECTED;
break;
case AIRPLAY_STATE_STREAMING:
state = OUTPUT_STATE_STREAMING;
break;
case AIRPLAY_STATE_TEARDOWN:
DPRINTF(E_LOG, L_RAOP, "Bug! session_status() called with transitional state (TEARDOWN)\n");
state = OUTPUT_STATE_STOPPED;
break;
default:
DPRINTF(E_LOG, L_RAOP, "Bug! Unhandled state in session_status(): %d\n", rs->state);
state = OUTPUT_STATE_FAILED;
}
outputs_cb(rs->callback_id, rs->device_id, state);
rs->callback_id = -1;
}
static struct airplay_master_session *
master_session_make(struct media_quality *quality, bool encrypt)
{
struct airplay_master_session *rms;
int ret;
// First check if we already have a suitable session
for (rms = airplay_master_sessions; rms; rms = rms->next)
{
if (encrypt == rms->encrypt && quality_is_equal(quality, &rms->rtp_session->quality))
return rms;
}
// Let's create a master session
ret = outputs_quality_subscribe(quality);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not subscribe to required audio quality (%d/%d/%d)\n", quality->sample_rate, quality->bits_per_sample, quality->channels);
return NULL;
}
CHECK_NULL(L_RAOP, rms = calloc(1, sizeof(struct airplay_master_session)));
rms->rtp_session = rtp_session_new(quality, RAOP_PACKET_BUFFER_SIZE, 0);
if (!rms->rtp_session)
{
outputs_quality_unsubscribe(quality);
free(rms);
return NULL;
}
rms->encrypt = encrypt;
rms->samples_per_packet = RAOP_SAMPLES_PER_PACKET;
rms->rawbuf_size = STOB(rms->samples_per_packet, quality->bits_per_sample, quality->channels);
rms->output_buffer_samples = OUTPUTS_BUFFER_DURATION * quality->sample_rate;
CHECK_NULL(L_RAOP, rms->rawbuf = malloc(rms->rawbuf_size));
CHECK_NULL(L_RAOP, rms->evbuf = evbuffer_new());
rms->next = airplay_master_sessions;
airplay_master_sessions = rms;
return rms;
}
static void
master_session_free(struct airplay_master_session *rms)
{
if (!rms)
return;
outputs_quality_unsubscribe(&rms->rtp_session->quality);
rtp_session_free(rms->rtp_session);
evbuffer_free(rms->evbuf);
free(rms->rawbuf);
free(rms);
}
static void
master_session_cleanup(struct airplay_master_session *rms)
{
struct airplay_master_session *s;
struct airplay_session *rs;
// First check if any other session is using the master session
for (rs = airplay_sessions; rs; rs=rs->next)
{
if (rs->master_session == rms)
return;
}
if (rms == airplay_master_sessions)
airplay_master_sessions = airplay_master_sessions->next;
else
{
for (s = airplay_master_sessions; s && (s->next != rms); s = s->next)
; /* EMPTY */
if (!s)
DPRINTF(E_WARN, L_RAOP, "WARNING: struct airplay_master_session not found in list; BUG!\n");
else
s->next = rms->next;
}
master_session_free(rms);
}
static void
session_free(struct airplay_session *rs)
{
if (!rs)
return;
if (rs->master_session)
master_session_cleanup(rs->master_session);
if (rs->ctrl)
{
evrtsp_connection_set_closecb(rs->ctrl, NULL, NULL);
evrtsp_connection_free(rs->ctrl);
}
if (rs->deferredev)
event_free(rs->deferredev);
if (rs->server_fd >= 0)
close(rs->server_fd);
pair_setup_free(rs->pair_setup_ctx);
pair_verify_free(rs->pair_verify_ctx);
pair_cipher_free(rs->control_cipher_ctx);
pair_cipher_free(rs->events_cipher_ctx);
free(rs->local_address);
free(rs->realm);
free(rs->nonce);
free(rs->session);
free(rs->address);
free(rs->devname);
free(rs);
}
static void
session_cleanup(struct airplay_session *rs)
{
struct airplay_session *s;
if (rs == airplay_sessions)
airplay_sessions = airplay_sessions->next;
else
{
for (s = airplay_sessions; s && (s->next != rs); s = s->next)
; /* EMPTY */
if (!s)
DPRINTF(E_WARN, L_RAOP, "WARNING: struct airplay_session not found in list; BUG!\n");
else
s->next = rs->next;
}
outputs_device_session_remove(rs->device_id);
session_free(rs);
}
static void
session_failure(struct airplay_session *rs)
{
/* Session failed, let our user know */
if (rs->state != AIRPLAY_STATE_PASSWORD)
rs->state = AIRPLAY_STATE_FAILED;
session_status(rs);
session_cleanup(rs);
}
static void
deferred_session_failure_cb(int fd, short what, void *arg)
{
struct airplay_session *rs = arg;
DPRINTF(E_DBG, L_RAOP, "Cleaning up failed session (deferred) on device '%s'\n", rs->devname);
session_failure(rs);
}
static void
deferred_session_failure(struct airplay_session *rs)
{
struct timeval tv;
if (rs->state != AIRPLAY_STATE_PASSWORD)
rs->state = AIRPLAY_STATE_FAILED;
evutil_timerclear(&tv);
evtimer_add(rs->deferredev, &tv);
}
static void
rtsp_close_cb(struct evrtsp_connection *evcon, void *arg)
{
struct airplay_session *rs = arg;
DPRINTF(E_LOG, L_RAOP, "Device '%s' closed RTSP connection\n", rs->devname);
deferred_session_failure(rs);
}
static void
session_success(struct airplay_session *rs)
{
session_status(rs);
session_cleanup(rs);
}
static void
session_connected(struct airplay_session *rs)
{
rs->state = AIRPLAY_STATE_CONNECTED;
session_status(rs);
}
static void
session_pair_success(struct airplay_session *rs)
{
if (rs->next_seq != AIRPLAY_SEQ_CONTINUE)
{
sequence_start(rs->next_seq, rs, NULL, "pair_success");
rs->next_seq = AIRPLAY_SEQ_CONTINUE;
return;
}
session_success(rs);
}
static int
session_connection_setup(struct airplay_session *rs, struct output_device *rd, int family)
{
char *address;
char *intf;
unsigned short port;
int ret;
rs->sa.ss.ss_family = family;
switch (family)
{
case AF_INET:
/* We always have the v4 services, so no need to check */
if (!rd->v4_address)
return -1;
address = rd->v4_address;
port = rd->v4_port;
rs->timing_svc = &timing_4svc;
rs->control_svc = &control_4svc;
ret = inet_pton(AF_INET, address, &rs->sa.sin.sin_addr);
break;
case AF_INET6:
if (!rd->v6_address || rd->v6_disabled || (timing_6svc.fd < 0) || (control_6svc.fd < 0))
return -1;
address = rd->v6_address;
port = rd->v6_port;
rs->timing_svc = &timing_6svc;
rs->control_svc = &control_6svc;
intf = strchr(address, '%');
if (intf)
*intf = '\0';
ret = inet_pton(AF_INET6, address, &rs->sa.sin6.sin6_addr);
if (intf)
{
*intf = '%';
intf++;
rs->sa.sin6.sin6_scope_id = if_nametoindex(intf);
if (rs->sa.sin6.sin6_scope_id == 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not find interface %s\n", intf);
ret = -1;
break;
}
}
break;
default:
return -1;
}
if (ret <= 0)
{
DPRINTF(E_LOG, L_RAOP, "Device '%s' has invalid address (%s) for %s\n", rd->name, address, (family == AF_INET) ? "ipv4" : "ipv6");
return -1;
}
rs->ctrl = evrtsp_connection_new(address, port);
if (!rs->ctrl)
{
DPRINTF(E_LOG, L_RAOP, "Could not create control connection to '%s' (%s)\n", rd->name, address);
return -1;
}
evrtsp_connection_set_base(rs->ctrl, evbase_player);
rs->address = strdup(address);
rs->family = family;
return 0;
}
static struct airplay_session *
session_make(struct output_device *rd, int callback_id)
{
struct airplay_session *rs;
struct airplay_extra *re;
int ret;
re = rd->extra_device_info;
CHECK_NULL(L_RAOP, rs = calloc(1, sizeof(struct airplay_session)));
CHECK_NULL(L_RAOP, rs->deferredev = evtimer_new(evbase_player, deferred_session_failure_cb, rs));
rs->devname = strdup(rd->name);
rs->volume = rd->volume;
rs->state = AIRPLAY_STATE_STOPPED;
rs->reqs_in_flight = 0;
rs->cseq = 1;
rs->device_id = rd->id;
rs->callback_id = callback_id;
rs->server_fd = -1;
rs->password = rd->password;
rs->supports_auth_setup = re->supports_auth_setup;
rs->wanted_metadata = re->wanted_metadata;
rs->next_seq = AIRPLAY_SEQ_CONTINUE;
rs->pair_type = PAIR_HOMEKIT_NORMAL;
#if AIRPLAY_USE_PAIRING_TRANSIENT
// requires_auth will be set if the device returned a 470 RTSP_CONNECTION_AUTH_REQUIRED
if (!rd->requires_auth && re->supports_pairing_transient)
rs->pair_type = PAIR_HOMEKIT_TRANSIENT;
#endif
switch (re->devtype)
{
case RAOP_DEV_APEX1_80211G:
rs->encrypt = 1;
rs->auth_quirk_itunes = 1;
break;
case RAOP_DEV_APEX2_80211N:
rs->encrypt = 1;
rs->auth_quirk_itunes = 0;
break;
case RAOP_DEV_APEX3_80211N:
rs->encrypt = 0;
rs->auth_quirk_itunes = 0;
break;
case RAOP_DEV_APPLETV:
rs->encrypt = 0;
rs->auth_quirk_itunes = 0;
break;
case RAOP_DEV_APPLETV4:
rs->encrypt = 0;
rs->auth_quirk_itunes = 0;
break;
default:
rs->encrypt = re->encrypt;
rs->auth_quirk_itunes = 0;
}
ret = session_connection_setup(rs, rd, AF_INET6);
if (ret < 0)
{
ret = session_connection_setup(rs, rd, AF_INET);
if (ret < 0)
goto error;
}
rs->master_session = master_session_make(&rd->quality, rs->encrypt);
if (!rs->master_session)
{
DPRINTF(E_LOG, L_RAOP, "Could not attach a master session for device '%s'\n", rd->name);
goto error;
}
// Attach to list of sessions
rs->next = airplay_sessions;
airplay_sessions = rs;
// rs is now the official device session
outputs_device_session_add(rd->id, rs);
return rs;
error:
session_free(rs);
return NULL;
}
/* ----------------------------- Metadata handling -------------------------- */
static void
airplay_metadata_free(struct airplay_metadata *rmd)
{
if (!rmd)
return;
if (rmd->metadata)
evbuffer_free(rmd->metadata);
if (rmd->artwork)
evbuffer_free(rmd->artwork);
free(rmd);
}
static void
airplay_metadata_purge(void)
{
if (!airplay_cur_metadata)
return;
airplay_metadata_free(airplay_cur_metadata->priv);
free(airplay_cur_metadata);
airplay_cur_metadata = NULL;
}
// *** Thread: worker ***
static void *
airplay_metadata_prepare(struct output_metadata *metadata)
{
struct db_queue_item *queue_item;
struct airplay_metadata *rmd;
struct evbuffer *tmp;
int ret;
queue_item = db_queue_fetch_byitemid(metadata->item_id);
if (!queue_item)
{
DPRINTF(E_LOG, L_RAOP, "Could not fetch queue item\n");
return NULL;
}
CHECK_NULL(L_RAOP, rmd = calloc(1, sizeof(struct airplay_metadata)));
CHECK_NULL(L_RAOP, rmd->artwork = evbuffer_new());
CHECK_NULL(L_RAOP, rmd->metadata = evbuffer_new());
CHECK_NULL(L_RAOP, tmp = evbuffer_new());
ret = artwork_get_item(rmd->artwork, queue_item->file_id, ART_DEFAULT_WIDTH, ART_DEFAULT_HEIGHT, 0);
if (ret < 0)
{
DPRINTF(E_INFO, L_RAOP, "Failed to retrieve artwork for file '%s'; no artwork will be sent\n", queue_item->path);
evbuffer_free(rmd->artwork);
rmd->artwork = NULL;
}
rmd->artwork_fmt = ret;
ret = dmap_encode_queue_metadata(rmd->metadata, tmp, queue_item);
evbuffer_free(tmp);
free_queue_item(queue_item, 0);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not encode file metadata; metadata will not be sent\n");
airplay_metadata_free(rmd);
return NULL;
}
return rmd;
}
static int
airplay_metadata_send_generic(struct airplay_session *rs, struct output_metadata *metadata, bool only_progress)
{
struct airplay_metadata *rmd = metadata->priv;
if (rs->wanted_metadata & RAOP_MD_WANTS_PROGRESS)
sequence_start(AIRPLAY_SEQ_SEND_PROGRESS, rs, metadata, "SET_PARAMETER (progress)");
if (!only_progress && (rs->wanted_metadata & RAOP_MD_WANTS_TEXT))
sequence_start(AIRPLAY_SEQ_SEND_TEXT, rs, metadata, "SET_PARAMETER (text)");
if (!only_progress && (rs->wanted_metadata & RAOP_MD_WANTS_ARTWORK) && rmd->artwork)
sequence_start(AIRPLAY_SEQ_SEND_ARTWORK, rs, metadata, "SET_PARAMETER (artwork)");
return 0;
}
static int
airplay_metadata_startup_send(struct airplay_session *rs)
{
if (!rs->wanted_metadata || !airplay_cur_metadata)
return 0;
airplay_cur_metadata->startup = true;
return airplay_metadata_send_generic(rs, airplay_cur_metadata, false);
}
static void
airplay_metadata_keep_alive_send(struct airplay_session *rs)
{
sequence_start(AIRPLAY_SEQ_FEEDBACK, rs, NULL, "keep_alive");
}
static void
airplay_metadata_send(struct output_metadata *metadata)
{
struct airplay_session *rs;
struct airplay_session *next;
int ret;
for (rs = airplay_sessions; rs; rs = next)
{
next = rs->next;
if (!(rs->state & AIRPLAY_STATE_F_CONNECTED) || !rs->wanted_metadata)
continue;
ret = airplay_metadata_send_generic(rs, metadata, false);
if (ret < 0)
{
session_failure(rs);
continue;
}
}
// Replace current metadata with the new stuff
airplay_metadata_purge();
airplay_cur_metadata = metadata;
}
/* ------------------------------ Volume handling --------------------------- */
static float
airplay_volume_from_pct(int volume, char *name)
{
float airplay_volume;
cfg_t *airplay;
int max_volume;
max_volume = RAOP_CONFIG_MAX_VOLUME;
airplay = cfg_gettsec(cfg, "airplay", name);
if (airplay)
max_volume = cfg_getint(airplay, "max_volume");
if ((max_volume < 1) || (max_volume > RAOP_CONFIG_MAX_VOLUME))
{
DPRINTF(E_LOG, L_RAOP, "Config has bad max_volume (%d) for device '%s', using default instead\n", max_volume, name);
max_volume = RAOP_CONFIG_MAX_VOLUME;
}
/* RAOP volume
* -144.0 is off
* 0 - 100 maps to -30.0 - 0
*/
if (volume > 0 && volume <= 100)
airplay_volume = -30.0 + ((float)max_volume * (float)volume * 30.0) / (100.0 * RAOP_CONFIG_MAX_VOLUME);
else
airplay_volume = -144.0;
return airplay_volume;
}
static int
airplay_volume_to_pct(struct output_device *rd, const char *volume)
{
float airplay_volume;
cfg_t *airplay;
int max_volume;
airplay_volume = atof(volume);
// Basic sanity check
if (airplay_volume == 0.0 && volume[0] != '0')
{
DPRINTF(E_LOG, L_RAOP, "RAOP device volume is invalid: '%s'\n", volume);
return -1;
}
max_volume = RAOP_CONFIG_MAX_VOLUME;
airplay = cfg_gettsec(cfg, "airplay", rd->name);
if (airplay)
max_volume = cfg_getint(airplay, "max_volume");
if ((max_volume < 1) || (max_volume > RAOP_CONFIG_MAX_VOLUME))
{
DPRINTF(E_LOG, L_RAOP, "Config has bad max_volume (%d) for device '%s', using default instead\n", max_volume, rd->name);
max_volume = RAOP_CONFIG_MAX_VOLUME;
}
// RAOP volume: -144.0 is off, -30.0 - 0 scaled by max_volume maps to 0 - 100
if (airplay_volume > -30.0 && airplay_volume <= 0.0)
return (int)(100.0 * (airplay_volume / 30.0 + 1.0) * RAOP_CONFIG_MAX_VOLUME / (float)max_volume);
else
return 0;
}
/* Volume in [0 - 100] */
static int
airplay_set_volume_one(struct output_device *device, int callback_id)
{
struct airplay_session *rs = device->session;
if (!rs || !(rs->state & AIRPLAY_STATE_F_CONNECTED))
return 0;
rs->volume = device->volume;
rs->callback_id = callback_id;
sequence_start(AIRPLAY_SEQ_SEND_VOLUME, rs, NULL, "set_volume_one");
return 1;
}
static void
airplay_keep_alive_timer_cb(int fd, short what, void *arg)
{
struct airplay_session *rs;
if (!airplay_sessions)
{
event_del(keep_alive_timer);
return;
}
for (rs = airplay_sessions; rs; rs = rs->next)
{
if (!(rs->state & AIRPLAY_STATE_F_CONNECTED))
continue;
airplay_metadata_keep_alive_send(rs);
}
evtimer_add(keep_alive_timer, &keep_alive_tv);
}
/* -------------------- Creation and sending of RTP packets ---------------- */
static int
packet_encrypt(uint8_t **out, size_t *out_len, struct rtp_packet *pkt, struct airplay_session *rs)
{
uint8_t authtag[16];
uint8_t nonce[12] = { 0 };
int nonce_offset = 4;
uint8_t *write_ptr;
int ret;
// Alloc so authtag and nonce can be appended
*out_len = pkt->data_len + sizeof(authtag) + sizeof(nonce) - nonce_offset;
*out = malloc(*out_len);
write_ptr = *out;
// Using seqnum as nonce not very secure, but means that when we resend
// packets they will be identical to the original
memcpy(nonce + nonce_offset, &pkt->seqnum, sizeof(pkt->seqnum));
// The RTP header is not encrypted
memcpy(write_ptr, pkt->header, pkt->header_len);
write_ptr = *out + pkt->header_len;
// Timestamp and SSRC are used as AAD = pkt->header + 4, len 8
ret = encrypt_chacha(write_ptr, pkt->payload, pkt->payload_len, rs->shared_secret, sizeof(rs->shared_secret), pkt->header + 4, 8, authtag, sizeof(authtag), nonce, sizeof(nonce));
if (ret < 0)
{
free(*out);
return -1;
}
write_ptr += pkt->payload_len;
memcpy(write_ptr, authtag, sizeof(authtag));
write_ptr += sizeof(authtag);
memcpy(write_ptr, nonce + nonce_offset, sizeof(nonce) - nonce_offset);
return 0;
}
static int
packet_send(struct airplay_session *rs, struct rtp_packet *pkt)
{
uint8_t *encrypted;
size_t encrypted_len;
int ret;
if (!rs)
return -1;
ret = packet_encrypt(&encrypted, &encrypted_len, pkt, rs);
if (ret < 0)
return -1;
ret = send(rs->server_fd, encrypted, encrypted_len, 0);
free(encrypted);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Send error for '%s': %s\n", rs->devname, strerror(errno));
// Can't free it right away, it would make the ->next in the calling
// master_session and session loops invalid
deferred_session_failure(rs);
return -1;
}
else if (ret != encrypted_len)
{
DPRINTF(E_WARN, L_RAOP, "Partial send (%d) for '%s'\n", ret, rs->devname);
return -1;
}
/* DPRINTF(E_DBG, L_RAOP, "RTP PACKET seqnum %u, rtptime %u, payload 0x%x, pktbuf_s %zu\n",
rs->master_session->rtp_session->seqnum,
rs->master_session->rtp_session->pos,
pkt->header[1],
rs->master_session->rtp_session->pktbuf_len
);
*/
return 0;
}
static void
control_packet_send(struct airplay_session *rs, struct rtp_packet *pkt)
{
int len;
int ret;
switch (rs->sa.ss.ss_family)
{
case AF_INET:
rs->sa.sin.sin_port = htons(rs->control_port);
len = sizeof(rs->sa.sin);
break;
case AF_INET6:
rs->sa.sin6.sin6_port = htons(rs->control_port);
len = sizeof(rs->sa.sin6);
break;
default:
DPRINTF(E_WARN, L_RAOP, "Unknown family %d\n", rs->sa.ss.ss_family);
return;
}
ret = sendto(rs->control_svc->fd, pkt->data, pkt->data_len, 0, &rs->sa.sa, len);
if (ret < 0)
DPRINTF(E_LOG, L_RAOP, "Could not send playback sync to device '%s': %s\n", rs->devname, strerror(errno));
}
static void
packets_resend(struct airplay_session *rs, uint16_t seqnum, int len)
{
struct rtp_session *rtp_session;
struct rtp_packet *pkt;
uint16_t s;
int i;
bool pkt_missing = false;
rtp_session = rs->master_session->rtp_session;
DPRINTF(E_DBG, L_RAOP, "Got retransmit request from '%s': seqnum %" PRIu16 " (len %d), last RTP session seqnum %" PRIu16 " (len %zu)\n",
rs->devname, seqnum, len, rtp_session->seqnum - 1, rtp_session->pktbuf_len);
// Note that seqnum may wrap around, so we don't use it for counting
for (i = 0, s = seqnum; i < len; i++, s++)
{
pkt = rtp_packet_get(rtp_session, s);
if (pkt)
packet_send(rs, pkt);
else
pkt_missing = true;
}
if (pkt_missing)
DPRINTF(E_WARN, L_RAOP, "Device '%s' retransmit request for seqnum %" PRIu16 " (len %d) is outside buffer range (last seqnum %" PRIu16 ", len %zu)\n",
rs->devname, seqnum, len, rtp_session->seqnum - 1, rtp_session->pktbuf_len);
}
static int
packets_send(struct airplay_master_session *rms)
{
struct rtp_packet *pkt;
struct airplay_session *rs;
pkt = rtp_packet_next(rms->rtp_session, ALAC_HEADER_LEN + rms->rawbuf_size, rms->samples_per_packet, RAOP_RTP_PAYLOADTYPE, 0);
alac_encode(pkt->payload, rms->rawbuf, rms->rawbuf_size);
for (rs = airplay_sessions; rs; rs = rs->next)
{
if (rs->master_session != rms)
continue;
// Device just joined
if (rs->state == AIRPLAY_STATE_CONNECTED)
{
pkt->header[1] = (1 << 7) | RAOP_RTP_PAYLOADTYPE;
packet_send(rs, pkt);
}
else if (rs->state == AIRPLAY_STATE_STREAMING)
{
pkt->header[1] = RAOP_RTP_PAYLOADTYPE;
packet_send(rs, pkt);
}
}
// Commits packet to retransmit buffer, and prepares the session for the next packet
rtp_packet_commit(rms->rtp_session, pkt);
return 0;
}
// Overview of rtptimes as they should be when starting a stream, and assuming
// the first rtptime (pos) is 88200:
// sync pkt: cur_pos = 0, rtptime = 88200
// audio pkt: rtptime = 88200
// RECORD: rtptime = 88200
// SET_PARAMETER text/artwork:
// rtptime = 88200
// SET_PARAMETER progress:
// progress = 72840/~88200/[len]
static inline void
timestamp_set(struct airplay_master_session *rms, struct timespec ts)
{
// The last write from the player had a timestamp which has been passed to
// this function as ts. This is the player clock, which is more precise than
// the actual clock because it gives us a calculated time reference, which is
// independent of how busy the thread is. We save that here, we need this for
// reference when sending sync packets and progress.
rms->cur_stamp.ts = ts;
// So what rtptime should be playing, i.e. coming out of the speaker, at time
// ts (which is normally "now")? Let's calculate by example:
// - we started playback with a rtptime (pos) of X
// - up until time ts we have received a 1000 samples from the player
// - rms->output_buffer_samples is configured to 400 samples
// -> we should be playing rtptime X + 600
//
// So how do we measure samples received from player? We know that from the
// pos, which says how much has been sent to the device, and from rms->evbuf,
// which is the unsent stuff being buffered:
// - received = (pos - X) + rms->evbuf_samples
//
// This means the rtptime is computed as:
// - rtptime = X + received - rms->output_buffer_samples
// -> rtptime = X + (pos - X) + rms->evbuf_samples - rms->out_buffer_samples
// -> rtptime = pos + rms->evbuf_samples - rms->output_buffer_samples
rms->cur_stamp.pos = rms->rtp_session->pos + rms->evbuf_samples - rms->output_buffer_samples;
}
static void
packets_sync_send(struct airplay_master_session *rms)
{
struct rtp_packet *sync_pkt;
struct airplay_session *rs;
struct timespec ts;
bool is_sync_time;
// Check if it is time send a sync packet to sessions that are already running
is_sync_time = rtp_sync_is_time(rms->rtp_session);
// Just used for logging, the clock shouldn't be too far from rms->cur_stamp.ts
clock_gettime(CLOCK_MONOTONIC, &ts);
for (rs = airplay_sessions; rs; rs = rs->next)
{
if (rs->master_session != rms)
continue;
// A device has joined and should get an init sync packet
if (rs->state == AIRPLAY_STATE_CONNECTED)
{
sync_pkt = rtp_sync_packet_next(rms->rtp_session, rms->cur_stamp, 0x90);
control_packet_send(rs, sync_pkt);
DPRINTF(E_DBG, L_RAOP, "Start sync packet sent to '%s': cur_pos=%" PRIu32 ", cur_ts=%ld.%09ld, clock=%ld.%09ld, rtptime=%" PRIu32 "\n",
rs->devname, rms->cur_stamp.pos, rms->cur_stamp.ts.tv_sec, rms->cur_stamp.ts.tv_nsec, ts.tv_sec, ts.tv_nsec, rms->rtp_session->pos);
}
else if (is_sync_time && rs->state == AIRPLAY_STATE_STREAMING)
{
sync_pkt = rtp_sync_packet_next(rms->rtp_session, rms->cur_stamp, 0x80);
control_packet_send(rs, sync_pkt);
}
}
}
/* ------------------------------ Time service ------------------------------ */
static void
airplay_timing_cb(int fd, short what, void *arg)
{
union sockaddr_all sa;
uint8_t req[32];
uint8_t res[32];
struct ntp_stamp recv_stamp;
struct ntp_stamp xmit_stamp;
struct airplay_service *svc;
int len;
int ret;
svc = (struct airplay_service *)arg;
ret = airplay_timing_get_clock_ntp(&recv_stamp);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't get receive timestamp\n");
goto readd;
}
len = sizeof(sa.ss);
ret = recvfrom(svc->fd, req, sizeof(req), 0, &sa.sa, (socklen_t *)&len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Error reading timing request: %s\n", strerror(errno));
goto readd;
}
if (ret != 32)
{
DPRINTF(E_DBG, L_RAOP, "Got timing request with size %d\n", ret);
goto readd;
}
if ((req[0] != 0x80) || (req[1] != 0xd2))
{
DPRINTF(E_LOG, L_RAOP, "Packet header doesn't match timing request (got 0x%02x%02x, expected 0x80d2)\n", req[0], req[1]);
goto readd;
}
memset(res, 0, sizeof(res));
/* Header */
res[0] = 0x80;
res[1] = 0xd3;
res[2] = req[2];
/* Copy client timestamp */
memcpy(res + 8, req + 24, 8);
/* Receive timestamp */
recv_stamp.sec = htobe32(recv_stamp.sec);
recv_stamp.frac = htobe32(recv_stamp.frac);
memcpy(res + 16, &recv_stamp.sec, 4);
memcpy(res + 20, &recv_stamp.frac, 4);
/* Transmit timestamp */
ret = airplay_timing_get_clock_ntp(&xmit_stamp);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't get transmit timestamp, falling back to receive timestamp\n");
/* Still better than failing altogether
* recv/xmit are close enough that it shouldn't matter much
*/
memcpy(res + 24, &recv_stamp.sec, 4);
memcpy(res + 28, &recv_stamp.frac, 4);
}
else
{
xmit_stamp.sec = htobe32(xmit_stamp.sec);
xmit_stamp.frac = htobe32(xmit_stamp.frac);
memcpy(res + 24, &xmit_stamp.sec, 4);
memcpy(res + 28, &xmit_stamp.frac, 4);
}
ret = sendto(svc->fd, res, sizeof(res), 0, &sa.sa, len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not send timing reply: %s\n", strerror(errno));
goto readd;
}
readd:
ret = event_add(svc->ev, NULL);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't re-add event for timing requests\n");
return;
}
}
static int
airplay_timing_start_one(struct airplay_service *svc, int family)
{
union sockaddr_all sa;
int on;
int len;
int ret;
int timing_port;
#ifdef SOCK_CLOEXEC
svc->fd = socket(family, SOCK_DGRAM | SOCK_CLOEXEC, 0);
#else
svc->fd = socket(family, SOCK_DGRAM, 0);
#endif
if (svc->fd < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't make timing socket: %s\n", strerror(errno));
return -1;
}
if (family == AF_INET6)
{
on = 1;
ret = setsockopt(svc->fd, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on));
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not set IPV6_V6ONLY on timing socket: %s\n", strerror(errno));
goto out_fail;
}
}
memset(&sa, 0, sizeof(union sockaddr_all));
sa.ss.ss_family = family;
timing_port = cfg_getint(cfg_getsec(cfg, "airplay_shared"), "timing_port");
switch (family)
{
case AF_INET:
sa.sin.sin_addr.s_addr = INADDR_ANY;
sa.sin.sin_port = htons(timing_port);
len = sizeof(sa.sin);
break;
case AF_INET6:
sa.sin6.sin6_addr = in6addr_any;
sa.sin6.sin6_port = htons(timing_port);
len = sizeof(sa.sin6);
break;
}
ret = bind(svc->fd, &sa.sa, len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't bind timing socket: %s\n", strerror(errno));
goto out_fail;
}
len = sizeof(sa.ss);
ret = getsockname(svc->fd, &sa.sa, (socklen_t *)&len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't get timing socket name: %s\n", strerror(errno));
goto out_fail;
}
switch (family)
{
case AF_INET:
svc->port = ntohs(sa.sin.sin_port);
DPRINTF(E_DBG, L_RAOP, "Timing IPv4 port: %d\n", svc->port);
break;
case AF_INET6:
svc->port = ntohs(sa.sin6.sin6_port);
DPRINTF(E_DBG, L_RAOP, "Timing IPv6 port: %d\n", svc->port);
break;
}
svc->ev = event_new(evbase_player, svc->fd, EV_READ, airplay_timing_cb, svc);
if (!svc->ev)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for airplay_service event\n");
goto out_fail;
}
event_add(svc->ev, NULL);
return 0;
out_fail:
close(svc->fd);
svc->fd = -1;
svc->port = 0;
return -1;
}
static void
airplay_timing_stop(void)
{
if (timing_4svc.ev)
event_free(timing_4svc.ev);
if (timing_6svc.ev)
event_free(timing_6svc.ev);
close(timing_4svc.fd);
timing_4svc.fd = -1;
timing_4svc.port = 0;
close(timing_6svc.fd);
timing_6svc.fd = -1;
timing_6svc.port = 0;
}
static int
airplay_timing_start(int v6enabled)
{
int ret;
if (v6enabled)
{
ret = airplay_timing_start_one(&timing_6svc, AF_INET6);
if (ret < 0)
DPRINTF(E_WARN, L_RAOP, "Could not start timing service on IPv6\n");
}
ret = airplay_timing_start_one(&timing_4svc, AF_INET);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not start timing service on IPv4\n");
airplay_timing_stop();
return -1;
}
return 0;
}
/* ----------------- Control service (retransmission and sync) ---------------*/
static void
airplay_control_cb(int fd, short what, void *arg)
{
char address[INET6_ADDRSTRLEN];
union sockaddr_all sa;
uint8_t req[8];
struct airplay_session *rs;
struct airplay_service *svc;
uint16_t seq_start;
uint16_t seq_len;
int len;
int ret;
svc = (struct airplay_service *)arg;
len = sizeof(sa.ss);
ret = recvfrom(svc->fd, req, sizeof(req), 0, &sa.sa, (socklen_t *)&len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Error reading control request: %s\n", strerror(errno));
goto readd;
}
if (ret != 8)
{
DPRINTF(E_DBG, L_RAOP, "Got control request with size %d\n", ret);
goto readd;
}
switch (sa.ss.ss_family)
{
case AF_INET:
if (svc != &control_4svc)
goto readd;
for (rs = airplay_sessions; rs; rs = rs->next)
{
if ((rs->sa.ss.ss_family == AF_INET)
&& (sa.sin.sin_addr.s_addr == rs->sa.sin.sin_addr.s_addr))
break;
}
if (!rs)
ret = (inet_ntop(AF_INET, &sa.sin.sin_addr.s_addr, address, sizeof(address)) != NULL);
break;
case AF_INET6:
if (svc != &control_6svc)
goto readd;
for (rs = airplay_sessions; rs; rs = rs->next)
{
if ((rs->sa.ss.ss_family == AF_INET6)
&& IN6_ARE_ADDR_EQUAL(&sa.sin6.sin6_addr, &rs->sa.sin6.sin6_addr))
break;
}
if (!rs)
ret = (inet_ntop(AF_INET6, &sa.sin6.sin6_addr.s6_addr, address, sizeof(address)) != NULL);
break;
default:
DPRINTF(E_LOG, L_RAOP, "Control svc: Unknown address family %d\n", sa.ss.ss_family);
goto readd;
}
if (!rs)
{
if (!ret)
DPRINTF(E_LOG, L_RAOP, "Control request from [error: %s]; not a RAOP client\n", strerror(errno));
else
DPRINTF(E_LOG, L_RAOP, "Control request from %s; not a RAOP client\n", address);
goto readd;
}
if ((req[0] != 0x80) || (req[1] != 0xd5))
{
DPRINTF(E_LOG, L_RAOP, "Packet header doesn't match retransmit request (got 0x%02x%02x, expected 0x80d5)\n", req[0], req[1]);
goto readd;
}
memcpy(&seq_start, req + 4, 2);
memcpy(&seq_len, req + 6, 2);
seq_start = be16toh(seq_start);
seq_len = be16toh(seq_len);
packets_resend(rs, seq_start, seq_len);
readd:
ret = event_add(svc->ev, NULL);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't re-add event for control requests\n");
return;
}
}
static int
airplay_control_start_one(struct airplay_service *svc, int family)
{
union sockaddr_all sa;
int on;
int len;
int ret;
int control_port;
#ifdef SOCK_CLOEXEC
svc->fd = socket(family, SOCK_DGRAM | SOCK_CLOEXEC, 0);
#else
svc->fd = socket(family, SOCK_DGRAM, 0);
#endif
if (svc->fd < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't make control socket: %s\n", strerror(errno));
return -1;
}
if (family == AF_INET6)
{
on = 1;
ret = setsockopt(svc->fd, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on));
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not set IPV6_V6ONLY on control socket: %s\n", strerror(errno));
goto out_fail;
}
}
memset(&sa, 0, sizeof(union sockaddr_all));
sa.ss.ss_family = family;
control_port = cfg_getint(cfg_getsec(cfg, "airplay_shared"), "control_port");
switch (family)
{
case AF_INET:
sa.sin.sin_addr.s_addr = INADDR_ANY;
sa.sin.sin_port = htons(control_port);
len = sizeof(sa.sin);
break;
case AF_INET6:
sa.sin6.sin6_addr = in6addr_any;
sa.sin6.sin6_port = htons(control_port);
len = sizeof(sa.sin6);
break;
}
ret = bind(svc->fd, &sa.sa, len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't bind control socket: %s\n", strerror(errno));
goto out_fail;
}
len = sizeof(sa.ss);
ret = getsockname(svc->fd, &sa.sa, (socklen_t *)&len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Couldn't get control socket name: %s\n", strerror(errno));
goto out_fail;
}
switch (family)
{
case AF_INET:
svc->port = ntohs(sa.sin.sin_port);
DPRINTF(E_DBG, L_RAOP, "Control IPv4 port: %d\n", svc->port);
break;
case AF_INET6:
svc->port = ntohs(sa.sin6.sin6_port);
DPRINTF(E_DBG, L_RAOP, "Control IPv6 port: %d\n", svc->port);
break;
}
svc->ev = event_new(evbase_player, svc->fd, EV_READ, airplay_control_cb, svc);
if (!svc->ev)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for control event\n");
goto out_fail;
}
event_add(svc->ev, NULL);
return 0;
out_fail:
close(svc->fd);
svc->fd = -1;
svc->port = 0;
return -1;
}
static void
airplay_control_stop(void)
{
if (control_4svc.ev)
event_free(control_4svc.ev);
if (control_6svc.ev)
event_free(control_6svc.ev);
close(control_4svc.fd);
control_4svc.fd = -1;
control_4svc.port = 0;
close(control_6svc.fd);
control_6svc.fd = -1;
control_6svc.port = 0;
}
static int
airplay_control_start(int v6enabled)
{
int ret;
if (v6enabled)
{
ret = airplay_control_start_one(&control_6svc, AF_INET6);
if (ret < 0)
DPRINTF(E_WARN, L_RAOP, "Could not start control service on IPv6\n");
}
ret = airplay_control_start_one(&control_4svc, AF_INET);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not start control service on IPv4\n");
airplay_control_stop();
return -1;
}
return 0;
}
/* ----------------------------- Event receiver ------------------------------*/
static void
event_channel_cb(int fd, short what, void *arg)
{
struct airplay_session *rs = arg;
ssize_t in_len;
int ret;
uint8_t in[4096]; //TODO
uint8_t *out;
size_t out_len = 0;
in_len = recv(fd, in, sizeof(in), 0);
if (in_len < 0)
DPRINTF(E_WARN, L_RAOP, "Possible disconnect from event channel from %s\n", rs->devname);
// TODO end session
if (in_len <= 0)
return;
DPRINTF(E_DBG, L_RAOP, "GOT AN EVENT, len was %zd\n", in_len);
if (in_len == sizeof(in))
return; // Longer than expected, give up
ret = pair_decrypt(&out, &out_len, in, in_len, rs->events_cipher_ctx);
if (ret < 0)
{
DPRINTF(E_DBG, L_RAOP, "Decryption error was: %s\n", pair_cipher_errmsg(rs->events_cipher_ctx));
return;
}
DHEXDUMP(E_DBG, L_RAOP, out, out_len, "Decrypted incoming event\n");
}
/* ----------------- Handlers for sending RAOP/RTSP requests ---------------- */
static int
payload_make_flush(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
struct airplay_master_session *rms = rs->master_session;
char buf[64];
int ret;
/* Restart sequence */
ret = snprintf(buf, sizeof(buf), "seq=%" PRIu16 ";rtptime=%u", rms->rtp_session->seqnum, rms->rtp_session->pos);
if ((ret < 0) || (ret >= sizeof(buf)))
{
DPRINTF(E_LOG, L_RAOP, "RTP-Info too big for buffer in FLUSH request\n");
return -1;
}
evrtsp_add_header(req->output_headers, "RTP-Info", buf);
return 0;
}
static int
payload_make_teardown(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
// Normally we update status when we get the response, but teardown is an
// exception because we want to stop writing to the device immediately
rs->state = AIRPLAY_STATE_TEARDOWN;
return 0;
}
static int
payload_make_set_volume(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
float raop_volume;
int ret;
raop_volume = airplay_volume_from_pct(rs->volume, rs->devname);
/* Don't let locales get in the way here */
/* We use -%d and -(int)raop_volume so -0.3 won't become 0.3 */
ret = evbuffer_add_printf(req->output_buffer, "volume: -%d.%06d\r\n", -(int)raop_volume, -(int)(1000000.0 * (raop_volume - (int)raop_volume)));
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for SET_PARAMETER payload (volume)\n");
return -1;
}
return 0;
}
static int
payload_make_send_progress(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
struct output_metadata *metadata = arg;
uint32_t start;
uint32_t display;
uint32_t pos;
uint32_t end;
int ret;
metadata_rtptimes_get(&start, &display, &pos, &end, rs->master_session, metadata);
ret = evbuffer_add_printf(req->output_buffer, "progress: %u/%u/%u\r\n", display, pos, end);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not build progress string for sending\n");
return -1;
}
ret = rtpinfo_header_add(req, rs, metadata);
if (ret < 0)
return -1;
return 0;
}
static int
payload_make_send_artwork(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
struct output_metadata *metadata = arg;
struct airplay_metadata *rmd = metadata->priv;
char *ctype;
uint8_t *buf;
size_t len;
int ret;
switch (rmd->artwork_fmt)
{
case ART_FMT_PNG:
ctype = "image/png";
break;
case ART_FMT_JPEG:
ctype = "image/jpeg";
break;
default:
DPRINTF(E_LOG, L_RAOP, "Unsupported artwork format %d\n", rmd->artwork_fmt);
return -1;
}
buf = evbuffer_pullup(rmd->artwork, -1);
len = evbuffer_get_length(rmd->artwork);
ret = evbuffer_add(req->output_buffer, buf, len);
if (ret != 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not copy artwork for sending\n");
return -1;
}
ret = rtpinfo_header_add(req, rs, metadata);
if (ret < 0)
return -1;
evrtsp_add_header(req->output_headers, "Content-Type", ctype);
return 0;
}
static int
payload_make_send_text(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
struct output_metadata *metadata = arg;
struct airplay_metadata *rmd = metadata->priv;
uint8_t *buf;
size_t len;
int ret;
buf = evbuffer_pullup(rmd->metadata, -1);
len = evbuffer_get_length(rmd->metadata);
ret = evbuffer_add(req->output_buffer, buf, len);
if (ret != 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not copy metadata for sending\n");
return -1;
}
ret = rtpinfo_header_add(req, rs, metadata);
if (ret < 0)
return -1;
return 0;
}
/*
Audio formats
Bit Value Type
2 0x4 PCM/8000/16/1
3 0x8 PCM/8000/16/2
4 0x10 PCM/16000/16/1
5 0x20 PCM/16000/16/2
6 0x40 PCM/24000/16/1
7 0x80 PCM/24000/16/2
8 0x100 PCM/32000/16/1
9 0x200 PCM/32000/16/2
10 0x400 PCM/44100/16/1
11 0x800 PCM/44100/16/2
12 0x1000 PCM/44100/24/1
13 0x2000 PCM/44100/24/2
14 0x4000 PCM/48000/16/1
15 0x8000 PCM/48000/16/2
16 0x10000 PCM/48000/24/1
17 0x20000 PCM/48000/24/2
18 0x40000 ALAC/44100/16/2
19 0x80000 ALAC/44100/24/2
20 0x100000 ALAC/48000/16/2
21 0x200000 ALAC/48000/24/2
22 0x400000 AAC-LC/44100/2
23 0x800000 AAC-LC/48000/2
24 0x1000000 AAC-ELD/44100/2
25 0x2000000 AAC-ELD/48000/2
26 0x4000000 AAC-ELD/16000/1
27 0x8000000 AAC-ELD/24000/1
28 0x10000000 OPUS/16000/1
29 0x20000000 OPUS/24000/1
30 0x40000000 OPUS/48000/1
31 0x80000000 AAC-ELD/44100/1
32 0x100000000 AAC-ELD/48000/1
*/
// {'streams': [{'audioFormat': 262144,
// 'audioMode': 'default',
// 'controlPort': 60242,
// 'ct': 2,
// 'isMedia': True,
// 'latencyMax': 88200,
// 'latencyMin': 11025,
// 'shk': b'\xdbc\x9b,\xdb\x15\x82\x19\x0b\xbf\xd3\xd0\x81\xc5\x7f7'
// b'\xaf\x7f\xb2l\xec\xca\xc8\xd0\x8d\x9d\x1d\xa8'
// b'\xda\xa2\xf7\xd0',
// 'spf': 352,
// 'supportsDynamicStreamID': True,
// 'type': 96}]}
static int
payload_make_setup_stream(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
plist_t root;
plist_t streams;
plist_t stream;
uint8_t *data;
size_t len;
int ret;
stream = plist_new_dict();
wplist_dict_add_uint(stream, "audioFormat", 262144); // 0x40000 ALAC/44100/16/2
wplist_dict_add_string(stream, "audioMode", "default");
wplist_dict_add_uint(stream, "controlPort", rs->control_svc->port);
wplist_dict_add_uint(stream, "ct", 2); // Compression type, 1 LPCM, 2 ALAC, 3 AAC, 4 AAC ELD, 32 OPUS
wplist_dict_add_bool(stream, "isMedia", true); // ?
wplist_dict_add_uint(stream, "latencyMax", 88200);
wplist_dict_add_uint(stream, "latencyMin", 11025);
wplist_dict_add_data(stream, "shk", rs->shared_secret, sizeof(rs->shared_secret));
wplist_dict_add_uint(stream, "spf", 352); // frames per packet
wplist_dict_add_uint(stream, "sr", RAOP_QUALITY_SAMPLE_RATE_DEFAULT); // sample rate
wplist_dict_add_uint(stream, "type", RAOP_RTP_PAYLOADTYPE); // RTP type, 0x60 = 96 real time, 103 buffered
wplist_dict_add_bool(stream, "supportsDynamicStreamID", false);
wplist_dict_add_uint(stream, "streamConnectionID", rs->session_id); // Hopefully fine since we have one stream per session
streams = plist_new_array();
plist_array_append_item(streams, stream);
root = plist_new_dict();
plist_dict_set_item(root, "streams", streams);
ret = wplist_to_bin(&data, &len, root);
plist_free(root);
if (ret < 0)
return -1;
evbuffer_add(req->output_buffer, data, len);
return 0;
}
static int
payload_make_setpeers(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
plist_t item;
uint8_t *data;
size_t len;
int ret;
plist_t root;
// TODO also have ipv6
root = plist_new_array();
item = plist_new_string(rs->address);
plist_array_append_item(root, item);
item = plist_new_string(rs->local_address);
plist_array_append_item(root, item);
ret = wplist_to_bin(&data, &len, root);
plist_free(root);
if (ret < 0)
return -1;
evbuffer_add(req->output_buffer, data, len);
return 0;
}
static int
payload_make_record(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
struct airplay_master_session *rms = rs->master_session;
char buf[64];
int ret;
evrtsp_add_header(req->output_headers, "X-Apple-ProtocolVersion", "1");
evrtsp_add_header(req->output_headers, "Range", "npt=0-");
// Start sequence: next sequence
ret = snprintf(buf, sizeof(buf), "seq=%" PRIu16 ";rtptime=%u", rms->rtp_session->seqnum, rms->rtp_session->pos);
if ((ret < 0) || (ret >= sizeof(buf)))
{
DPRINTF(E_LOG, L_RAOP, "RTP-Info too big for buffer in RECORD request\n");
return -1;
}
evrtsp_add_header(req->output_headers, "RTP-Info", buf);
DPRINTF(E_DBG, L_RAOP, "RTP-Info is %s\n", buf);
return 0;
}
// {'deviceID': '11:22:33:44:55:66',
// 'eiv': b'=o\xa0\xc24\xcd\xee\xcb9\x99~l\x140\x08\x9c',
// 'ekey': b'\x08\x90x\xa6\x0e\x87$C\x88l\xc1MS[Q\xaf',
// 'et': 0,
// 'groupContainsGroupLeader': False,
// 'groupUUID': '67EAD1FA-7EAB-4810-82F7-A9132FD2D0BB',
// 'isMultiSelectAirPlay': True,
// 'macAddress': '11:22:33:44:55:68',
// 'model': 'iPhone10,6',
// 'name': 'iPXema',
// 'osBuildVersion': '17B111',
// 'osName': 'iPhone OS',
// 'osVersion': '13.2.3',
// 'senderSupportsRelay': True,
// 'sessionUUID': '3195C737-1E6E-4487-BECB-4D287B7C7626',
// 'sourceVersion': '409.16',
// 'timingPeerInfo': {'Addresses': ['192.168.1.86', 'fe80::473:74c7:28a7:3bee'],
// 'ID': '67EAD1FA-7EAB-4810-82F7-A9132FD2D0BB',
// 'SupportsClockPortMatchingOverride': True},
// 'timingPeerList': [{'Addresses': ['192.168.1.86', 'fe80::473:74c7:28a7:3bee'],
// 'ID': '67EAD1FA-7EAB-4810-82F7-A9132FD2D0BB',
// 'SupportsClockPortMatchingOverride': True}],
// 'timingProtocol': 'PTP'}
static int
payload_make_setup_session(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
plist_t root;
// plist_t timingpeerinfo;
// plist_t timingpeerlist;
plist_t addresses;
plist_t address;
uint8_t *data;
size_t len;
int ret;
ret = session_url_set(rs);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not make session url for device '%s'\n", rs->devname);
return -1;
}
address = plist_new_string(rs->local_address);
addresses = plist_new_array();
plist_array_append_item(addresses, address);
/* timingpeerinfo = plist_new_dict();
plist_dict_set_item(timingpeerinfo, "Addresses", addresses);
wplist_dict_add_string(timingpeerinfo, "ID", "67EAD1FA-7EAB-4810-82F7-A9132FD2D0BB");
wplist_dict_add_bool(timingpeerinfo, "SupportsClockPortMatchingOverride", false);
*/
/* timingpeerlist = plist_new_dict();
plist_dict_set_item(timingpeerlist, "Addresses", addresses);
wplist_dict_add_string(timingpeerlist, "ID", "67EAD1FA-7EAB-4810-82F7-A9132FD2D0BB");
wplist_dict_add_bool(timingpeerlist, "SupportsClockPortMatchingOverride", false);
*/
root = plist_new_dict();
// wplist_dict_add_string(root, "deviceID", "11:22:33:44:55:66");
// wplist_dict_add_data(root, "eiv", airplay_aes_iv, sizeof(airplay_aes_iv));
// wplist_dict_add_data(root, "ekey", airplay_aes_key, sizeof(airplay_aes_key));
// wplist_dict_add_uint(root, "et", 0); // No encryption?
// wplist_dict_add_bool(root, "groupContainsGroupLeader", true);
// wplist_dict_add_string(root, "groupUUID", "3195C737-1E6E-4487-BECB-4D287B7C1234");
// wplist_dict_add_bool(root, "internalBuild", false);
// wplist_dict_add_bool(root, "isMultiSelectAirPlay", true);
// wplist_dict_add_string(root, "macAddress", "00:0c:29:f6:4a:f9");
// wplist_dict_add_string(root, "model", "iPhone10,4");
// wplist_dict_add_string(root, "osBuildVersion", "18B92");
// wplist_dict_add_string(root, "osName", "iPhone OS");
// wplist_dict_add_string(root, "osVersion", "14.2");
// wplist_dict_add_bool(root, "senderSupportsRelay", true);
wplist_dict_add_string(root, "sessionUUID", "3195C737-1E6E-4487-BECB-4D287B7C7626");
// wplist_dict_add_string(root, "sourceVersion", "525.38.2");
// plist_dict_set_item(root, "timingPeerInfo", timingpeerinfo); // only for PTP timing?
// plist_dict_set_item(root, "timingPeerList", timingpeerlist); // only for PTP timing?
wplist_dict_add_uint(root, "timingPort", rs->timing_svc->port);
wplist_dict_add_string(root, "timingProtocol", "NTP"); // If set to "None" then an ATV4 will not respond to stream SETUP request
// wplist_dict_add_string(root, "timingProtocol", "None");
ret = wplist_to_bin(&data, &len, root);
plist_free(root);
if (ret < 0)
return -1;
evbuffer_add(req->output_buffer, data, len);
return 0;
}
/*
The purpose of auth-setup is to authenticate the device and to exchange keys
for encryption. We don't do that, but some AirPlay 2 speakers (Sonos beam,
Airport Express fw 7.8) require this step anyway, otherwise we get a 403 to
our ANNOUNCE. So we do it with a flag for no encryption, and without actually
authenticating the device.
Good to know (source Apple's MFi Accessory Interface Specification):
- Curve25519 Elliptic-Curve Diffie-Hellman technology for key exchange
- RSA for signing and verifying and AES-128 in counter mode for encryption
- We start by sending a Curve25519 public key + no encryption flag
- The device responds with public key, MFi certificate and a signature, which
is created by the device signing the two public keys with its RSA private
key and then encrypting the result with the AES master key derived from the
Curve25519 shared secret (generated from device private key and our public
key)
- The AES key derived from the Curve25519 shared secret can then be used to
encrypt future content
- New keys should be generated for each authentication attempt, but we don't
do that because we don't really use this + it adds a libsodium dependency
Since we don't do auth or encryption, we currently just ignore the reponse.
*/
/*
static int
payload_make_auth_setup(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
if (!(rs->supports_post && rs->supports_auth_setup))
return 1; // skip this request
// Flag for no encryption. 0x10 may mean encryption.
evbuffer_add(req->output_buffer, "\x01", 1);
evbuffer_add(req->output_buffer, airplay_auth_setup_pubkey, sizeof(airplay_auth_setup_pubkey) - 1);
return 0;
}
*/
static int
payload_make_pin_start(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
DPRINTF(E_LOG, L_RAOP, "Starting device pairing for '%s', go to the web interface and enter PIN\n", rs->devname);
return 0;
}
static int
payload_make_pair_generic(int step, struct evrtsp_request *req, struct airplay_session *rs)
{
uint8_t *body;
uint32_t len;
const char *errmsg;
switch (step)
{
case 1:
body = pair_setup_request1(&len, rs->pair_setup_ctx);
errmsg = pair_setup_errmsg(rs->pair_setup_ctx);
break;
case 2:
body = pair_setup_request2(&len, rs->pair_setup_ctx);
errmsg = pair_setup_errmsg(rs->pair_setup_ctx);
break;
case 3:
body = pair_setup_request3(&len, rs->pair_setup_ctx);
errmsg = pair_setup_errmsg(rs->pair_setup_ctx);
break;
case 4:
body = pair_verify_request1(&len, rs->pair_verify_ctx);
errmsg = pair_verify_errmsg(rs->pair_verify_ctx);
break;
case 5:
body = pair_verify_request2(&len, rs->pair_verify_ctx);
errmsg = pair_verify_errmsg(rs->pair_verify_ctx);
break;
default:
body = NULL;
errmsg = "Bug! Bad step number";
}
if (!body)
{
DPRINTF(E_LOG, L_RAOP, "Verification step %d request error: %s\n", step, errmsg);
return -1;
}
evbuffer_add(req->output_buffer, body, len);
free(body);
// Required!!
if (rs->pair_type == PAIR_HOMEKIT_NORMAL)
evrtsp_add_header(req->output_headers, "X-Apple-HKP", "3");
else if (rs->pair_type == PAIR_HOMEKIT_TRANSIENT)
evrtsp_add_header(req->output_headers, "X-Apple-HKP", "4");
return 0;
}
static int
payload_make_pair_setup1(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
char *pin = arg;
if (pin)
rs->pair_type = PAIR_HOMEKIT_NORMAL;
rs->pair_setup_ctx = pair_setup_new(rs->pair_type, pin, pair_device_id);
if (!rs->pair_setup_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for verification setup context\n");
return -1;
}
rs->state = AIRPLAY_STATE_PASSWORD;
return payload_make_pair_generic(1, req, rs);
}
static int
payload_make_pair_setup2(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
return payload_make_pair_generic(2, req, rs);
}
static int
payload_make_pair_setup3(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
return payload_make_pair_generic(3, req, rs);
}
static int
payload_make_pair_verify1(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
struct output_device *device;
device = outputs_device_get(rs->device_id);
if (!device)
return -1;
rs->pair_verify_ctx = pair_verify_new(rs->pair_type, device->auth_key, pair_device_id);
if (!rs->pair_verify_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Out of memory for verification verify context\n");
return -1;
}
return payload_make_pair_generic(4, req, rs);
}
static int
payload_make_pair_verify2(struct evrtsp_request *req, struct airplay_session *rs, void *arg)
{
return payload_make_pair_generic(5, req, rs);
}
/* ------------------------------ Session startup --------------------------- */
static int
device_connect(struct airplay_session *rs, unsigned short port, int type)
{
int len;
int fd;
int ret;
DPRINTF(E_DBG, L_RAOP, "Connecting to %s (family=%d), port %u\n", rs->address, rs->family, port);
#ifdef SOCK_CLOEXEC
fd = socket(rs->sa.ss.ss_family, type | SOCK_CLOEXEC, 0);
#else
fd = socket(rs->sa.ss.ss_family, type, 0);
#endif
if (fd < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not create socket: %s\n", strerror(errno));
return -1;
}
switch (rs->sa.ss.ss_family)
{
case AF_INET:
rs->sa.sin.sin_port = htons(port);
len = sizeof(rs->sa.sin);
break;
case AF_INET6:
rs->sa.sin6.sin6_port = htons(port);
len = sizeof(rs->sa.sin6);
break;
default:
DPRINTF(E_WARN, L_RAOP, "Unknown family %d\n", rs->sa.ss.ss_family);
close(fd);
return -1;
}
ret = connect(fd, &rs->sa.sa, len);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "connect() to [%s]:%u failed: %s\n", rs->address, port, strerror(errno));
close(fd);
return -1;
}
return fd;
}
static void
start_failure(struct airplay_session *rs)
{
// Tear down the connection
sequence_start(AIRPLAY_SEQ_FAILURE, rs, NULL, "startup_failure");
}
static void
start_retry(struct airplay_session *rs)
{
struct output_device *device;
int callback_id = rs->callback_id;
device = outputs_device_get(rs->device_id);
if (!device || !rs->session)
{
session_failure(rs);
return;
}
// Some devices don't seem to work with ipv6, so if the error wasn't a hard
// failure (bad password) we fall back to ipv4 and flag device as bad for ipv6
if (rs->family != AF_INET6 || (rs->state & AIRPLAY_STATE_F_FAILED))
{
session_failure(rs);
return;
}
// This flag is permanent and will not be overwritten by mdns advertisements
device->v6_disabled = 1;
// Drop session, try again with ipv4
session_cleanup(rs);
airplay_device_start(device, callback_id);
}
/* ---------------------------- RTSP response handlers ---------------------- */
static enum airplay_seq_type
response_handler_pin_start(struct evrtsp_request *req, struct airplay_session *rs)
{
rs->state = AIRPLAY_STATE_PASSWORD;
return AIRPLAY_SEQ_CONTINUE; // TODO before we reported failure since device is locked
}
static enum airplay_seq_type
response_handler_record(struct evrtsp_request *req, struct airplay_session *rs)
{
const char *param;
/* Audio latency */
param = evrtsp_find_header(req->input_headers, "Audio-Latency");
if (!param)
DPRINTF(E_INFO, L_RAOP, "RECORD reply from '%s' did not have an Audio-Latency header\n", rs->devname);
else
DPRINTF(E_DBG, L_RAOP, "RAOP audio latency is %s\n", param);
rs->state = AIRPLAY_STATE_RECORD;
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_setup_stream(struct evrtsp_request *req, struct airplay_session *rs)
{
plist_t response;
plist_t streams;
plist_t stream;
plist_t item;
uint64_t uintval;
int ret;
ret = wplist_from_evbuf(&response, req->input_buffer);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not parse plist from '%s'\n", rs->devname);
return AIRPLAY_SEQ_ABORT;
}
streams = plist_dict_get_item(response, "streams");
if (!streams)
{
DPRINTF(E_LOG, L_RAOP, "Could not find streams item in response from '%s'\n", rs->devname);
goto error;
}
stream = plist_array_get_item(streams, 0);
if (!stream)
{
DPRINTF(E_LOG, L_RAOP, "Could not find stream item in response from '%s'\n", rs->devname);
goto error;
}
item = plist_dict_get_item(stream, "dataPort");
if (item)
{
plist_get_uint_val(item, &uintval);
rs->data_port = uintval;
}
item = plist_dict_get_item(stream, "controlPort");
if (item)
{
plist_get_uint_val(item, &uintval);
rs->control_port = uintval;
}
if (rs->data_port == 0 || rs->control_port == 0)
{
DPRINTF(E_LOG, L_RAOP, "Missing port number in reply from '%s' (d=%u, c=%u)\n", rs->devname, rs->data_port, rs->control_port);
goto error;
}
DPRINTF(E_DBG, L_RAOP, "Negotiated AirTunes v2 UDP streaming session %s; ports d=%u c=%u t=%u e=%u\n", rs->session, rs->data_port, rs->control_port, rs->timing_port, rs->events_port);
rs->server_fd = device_connect(rs, rs->data_port, SOCK_DGRAM);
if (rs->server_fd < 0)
{
DPRINTF(E_WARN, L_RAOP, "Could not connect to data port\n");
goto error;
}
// Reverse connection, used to receive playback events from device
rs->events_fd = device_connect(rs, rs->events_port, SOCK_STREAM);
if (rs->events_fd < 0)
{
DPRINTF(E_WARN, L_RAOP, "Could not connect to '%s' events port %u, proceeding anyway\n", rs->devname, rs->events_port);
}
else
{
struct event *ev = event_new(evbase_player, rs->events_fd, EV_READ | EV_PERSIST, event_channel_cb, rs); // TODO, possibly use evrtsp instead
event_add(ev, NULL);
}
rs->state = AIRPLAY_STATE_SETUP;
plist_free(response);
return AIRPLAY_SEQ_CONTINUE;
error:
plist_free(response);
return AIRPLAY_SEQ_ABORT;
}
static enum airplay_seq_type
response_handler_volume_start(struct evrtsp_request *req, struct airplay_session *rs)
{
int ret;
ret = airplay_metadata_startup_send(rs); // TODO Should this be added to the startup sequence?
if (ret < 0)
return AIRPLAY_SEQ_ABORT;
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_setup_session(struct evrtsp_request *req, struct airplay_session *rs)
{
plist_t response;
plist_t item;
uint64_t uintval;
int ret;
ret = wplist_from_evbuf(&response, req->input_buffer);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not parse plist from '%s'\n", rs->devname);
return AIRPLAY_SEQ_ABORT;
}
item = plist_dict_get_item(response, "eventPort");
if (item)
{
plist_get_uint_val(item, &uintval);
rs->events_port = uintval;
}
item = plist_dict_get_item(response, "timingPort");
if (item)
{
plist_get_uint_val(item, &uintval);
rs->timing_port = uintval;
}
if (rs->events_port == 0)
{
DPRINTF(E_LOG, L_RAOP, "SETUP reply is missing event port\n");
goto error;
}
plist_free(response);
return AIRPLAY_SEQ_CONTINUE;
error:
plist_free(response);
return AIRPLAY_SEQ_ABORT;
}
static enum airplay_seq_type
response_handler_flush(struct evrtsp_request *req, struct airplay_session *rs)
{
rs->state = AIRPLAY_STATE_CONNECTED;
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_teardown(struct evrtsp_request *req, struct airplay_session *rs)
{
rs->state = AIRPLAY_STATE_STOPPED;
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_teardown_failure(struct evrtsp_request *req, struct airplay_session *rs)
{
if (rs->state != AIRPLAY_STATE_PASSWORD)
rs->state = AIRPLAY_STATE_FAILED;
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_options_generic(struct evrtsp_request *req, struct airplay_session *rs)
{
struct output_device *device;
const char *param;
int ret;
if ((req->response_code != RTSP_OK) && (req->response_code != RTSP_UNAUTHORIZED) && (req->response_code != RTSP_FORBIDDEN))
{
DPRINTF(E_LOG, L_RAOP, "OPTIONS request failed '%s' (%s): %d %s\n", rs->devname, rs->address, req->response_code, req->response_code_line);
goto error;
}
if (req->response_code == RTSP_UNAUTHORIZED)
{
if (rs->req_has_auth)
{
DPRINTF(E_LOG, L_RAOP, "Bad password for device '%s' (%s)\n", rs->devname, rs->address);
rs->state = AIRPLAY_STATE_PASSWORD;
goto error;
}
ret = response_header_auth_parse(rs, req);
if (ret < 0)
{
goto error;
}
return AIRPLAY_SEQ_START_RERUN;
}
if (req->response_code == RTSP_FORBIDDEN)
{
device = outputs_device_get(rs->device_id);
if (!device)
goto error;
device->requires_auth = 1;
return AIRPLAY_SEQ_PIN_START;
}
param = evrtsp_find_header(req->input_headers, "Public");
if (param)
rs->supports_post = (strstr(param, "POST") != NULL);
else
DPRINTF(E_DBG, L_RAOP, "Could not find 'Public' header in OPTIONS reply from '%s' (%s)\n", rs->devname, rs->address);
rs->state = AIRPLAY_STATE_OPTIONS;
return AIRPLAY_SEQ_CONTINUE;
error:
return AIRPLAY_SEQ_ABORT;
}
static enum airplay_seq_type
response_handler_options_probe(struct evrtsp_request *req, struct airplay_session *rs)
{
return response_handler_options_generic(req, rs);
}
static enum airplay_seq_type
response_handler_options_start(struct evrtsp_request *req, struct airplay_session *rs)
{
enum airplay_seq_type seq_type;
seq_type = response_handler_options_generic(req, rs);
if (seq_type != AIRPLAY_SEQ_CONTINUE)
return seq_type;
return AIRPLAY_SEQ_START_AP2;
}
static enum airplay_seq_type
response_handler_pair_generic(int step, struct evrtsp_request *req, struct airplay_session *rs)
{
uint8_t *response;
const char *errmsg;
size_t len;
int ret;
response = evbuffer_pullup(req->input_buffer, -1);
len = evbuffer_get_length(req->input_buffer);
switch (step)
{
case 1:
ret = pair_setup_response1(rs->pair_setup_ctx, response, len);
errmsg = pair_setup_errmsg(rs->pair_setup_ctx);
break;
case 2:
ret = pair_setup_response2(rs->pair_setup_ctx, response, len);
errmsg = pair_setup_errmsg(rs->pair_setup_ctx);
break;
case 3:
ret = pair_setup_response3(rs->pair_setup_ctx, response, len);
errmsg = pair_setup_errmsg(rs->pair_setup_ctx);
break;
case 4:
ret = pair_verify_response1(rs->pair_verify_ctx, response, len);
errmsg = pair_verify_errmsg(rs->pair_verify_ctx);
break;
case 5:
ret = pair_verify_response2(rs->pair_verify_ctx, response, len);
errmsg = pair_verify_errmsg(rs->pair_verify_ctx);
break;
default:
ret = -1;
errmsg = "Bug! Bad step number";
}
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Pairing step %d response from '%s' error: %s\n", step, rs->devname, errmsg);
DHEXDUMP(E_DBG, L_RAOP, response, len, "Raw response");
return AIRPLAY_SEQ_ABORT;
}
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_pair_setup1(struct evrtsp_request *req, struct airplay_session *rs)
{
struct output_device *device;
if (rs->pair_type == PAIR_HOMEKIT_TRANSIENT && req->response_code == RTSP_CONNECTION_AUTH_REQUIRED)
{
device = outputs_device_get(rs->device_id);
if (!device)
return AIRPLAY_SEQ_ABORT;
device->requires_auth = 1; // FIXME might be reset by mdns announcement
rs->pair_type = PAIR_HOMEKIT_NORMAL;
return AIRPLAY_SEQ_PIN_START;
}
return response_handler_pair_generic(1, req, rs);
}
static enum airplay_seq_type
response_handler_pair_setup2(struct evrtsp_request *req, struct airplay_session *rs)
{
enum airplay_seq_type seq_type;
const uint8_t *shared_secret;
size_t shared_secret_len;
int ret;
seq_type = response_handler_pair_generic(2, req, rs);
if (seq_type != AIRPLAY_SEQ_CONTINUE)
return seq_type;
if (rs->pair_type != PAIR_HOMEKIT_TRANSIENT)
return seq_type;
ret = pair_setup_result(NULL, &shared_secret, &shared_secret_len, rs->pair_setup_ctx);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Transient setup result error: %s\n", pair_setup_errmsg(rs->pair_setup_ctx));
goto error;
}
if (shared_secret_len < sizeof(rs->shared_secret)) // We expect 64 bytes, and rs->shared_secret is 32 bytes
{
DPRINTF(E_LOG, L_RAOP, "Transient setup result error: Unexpected key length (%zu)\n", shared_secret_len);
goto error;
}
// Copy the first 32 bytes while be used later for encrypting audio payload
memcpy(rs->shared_secret, shared_secret, sizeof(rs->shared_secret));
rs->control_cipher_ctx = pair_cipher_new(rs->pair_type, 0, shared_secret, shared_secret_len);
if (!rs->control_cipher_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Could not create control ciphering context\n");
goto error;
}
rs->events_cipher_ctx = pair_cipher_new(rs->pair_type, 1, shared_secret, shared_secret_len);
if (!rs->events_cipher_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Could not create events ciphering context\n");
goto error;
}
evrtsp_connection_set_ciphercb(rs->ctrl, rtsp_cipher, rs);
DPRINTF(E_INFO, L_RAOP, "Transient setup of '%s' completed succesfully, now using encrypted mode\n", rs->devname);
rs->state = AIRPLAY_STATE_STARTUP;
return AIRPLAY_SEQ_CONTINUE;
error:
rs->state = AIRPLAY_STATE_FAILED;
return AIRPLAY_SEQ_ABORT;
}
static enum airplay_seq_type
response_handler_pair_setup3(struct evrtsp_request *req, struct airplay_session *rs)
{
struct output_device *device;
const char *authorization_key;
enum airplay_seq_type seq_type;
int ret;
seq_type = response_handler_pair_generic(3, req, rs);
if (seq_type != AIRPLAY_SEQ_CONTINUE)
return seq_type;
ret = pair_setup_result(&authorization_key, NULL, NULL, rs->pair_setup_ctx);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Pair setup result error: %s\n", pair_setup_errmsg(rs->pair_setup_ctx));
return AIRPLAY_SEQ_ABORT;
}
DPRINTF(E_LOG, L_RAOP, "Pair setup stage complete, saving authorization key\n");
device = outputs_device_get(rs->device_id);
if (!device)
return AIRPLAY_SEQ_ABORT;
free(device->auth_key);
device->auth_key = strdup(authorization_key);
// A blocking db call... :-~
db_speaker_save(device);
// No longer AIRPLAY_STATE_PASSWORD
rs->state = AIRPLAY_STATE_STOPPED;
return AIRPLAY_SEQ_CONTINUE;
}
static enum airplay_seq_type
response_handler_pair_verify1(struct evrtsp_request *req, struct airplay_session *rs)
{
struct output_device *device;
enum airplay_seq_type seq_type;
seq_type = response_handler_pair_generic(4, req, rs);
if (seq_type != AIRPLAY_SEQ_CONTINUE)
{
rs->state = AIRPLAY_STATE_PASSWORD;
device = outputs_device_get(rs->device_id);
if (!device)
return AIRPLAY_SEQ_ABORT;
// Clear auth_key, the device did not accept it
free(device->auth_key);
device->auth_key = NULL;
return AIRPLAY_SEQ_ABORT;
}
return seq_type;
}
static enum airplay_seq_type
response_handler_pair_verify2(struct evrtsp_request *req, struct airplay_session *rs)
{
struct output_device *device;
enum airplay_seq_type seq_type;
const uint8_t *shared_secret;
size_t shared_secret_len;
int ret;
seq_type = response_handler_pair_generic(5, req, rs);
if (seq_type != AIRPLAY_SEQ_CONTINUE)
goto error;
ret = pair_verify_result(&shared_secret, &shared_secret_len, rs->pair_verify_ctx);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Pair verify result error: %s\n", pair_verify_errmsg(rs->pair_verify_ctx));
goto error;
}
if (sizeof(rs->shared_secret) != shared_secret_len)
{
DPRINTF(E_LOG, L_RAOP, "Pair verify result error: Unexpected key length (%zu)\n", shared_secret_len);
goto error;
}
memcpy(rs->shared_secret, shared_secret, shared_secret_len);
rs->control_cipher_ctx = pair_cipher_new(rs->pair_type, 0, rs->shared_secret, sizeof(rs->shared_secret));
if (!rs->control_cipher_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Could not create control ciphering context\n");
goto error;
}
rs->events_cipher_ctx = pair_cipher_new(rs->pair_type, 1, rs->shared_secret, sizeof(rs->shared_secret));
if (!rs->events_cipher_ctx)
{
DPRINTF(E_LOG, L_RAOP, "Could not create events ciphering context\n");
goto error;
}
evrtsp_connection_set_ciphercb(rs->ctrl, rtsp_cipher, rs);
DPRINTF(E_INFO, L_RAOP, "Pairing of '%s' completed succesfully, now using encrypted mode\n", rs->devname);
rs->state = AIRPLAY_STATE_STARTUP;
return AIRPLAY_SEQ_CONTINUE;
error:
device = outputs_device_get(rs->device_id);
if (!device)
return AIRPLAY_SEQ_ABORT;
// Clear auth_key, the device did not accept it, or some other unexpected error
free(device->auth_key);
device->auth_key = NULL;
rs->state = AIRPLAY_STATE_PASSWORD;
return AIRPLAY_SEQ_ABORT;
}
/* ---------------------- Request/response sequence control ----------------- */
/*
* Request queueing HOWTO
*
* Sending:
* - increment rs->reqs_in_flight
* - set evrtsp connection closecb to NULL
*
* Request callback:
* - decrement rs->reqs_in_flight first thing, even if the callback is
* called for error handling (req == NULL or HTTP error code)
* - if rs->reqs_in_flight == 0, setup evrtsp connection closecb
*
* When a request fails, the whole RAOP session is declared failed and
* torn down by calling session_failure(), even if there are requests
* queued on the evrtsp connection. There is no reason to think pending
* requests would work out better than the one that just failed and recovery
* would be tricky to get right.
*
* evrtsp behaviour with queued requests:
* - request callback is called with req == NULL to indicate a connection
* error; if there are several requests queued on the connection, this can
* happen for each request if the connection isn't destroyed
* - the connection is reset, and the closecb is called if the connection was
* previously connected. There is no closecb set when there are requests in
* flight
*/
static struct airplay_seq_definition airplay_seq_definition[] =
{
{ AIRPLAY_SEQ_START, NULL, start_retry },
{ AIRPLAY_SEQ_START_RERUN, NULL, start_retry },
{ AIRPLAY_SEQ_START_AP2, session_connected, start_failure },
{ AIRPLAY_SEQ_PROBE, session_success, session_failure },
{ AIRPLAY_SEQ_FLUSH, session_status, session_failure },
{ AIRPLAY_SEQ_STOP, session_success, session_failure },
{ AIRPLAY_SEQ_FAILURE, session_success, session_failure},
{ AIRPLAY_SEQ_PIN_START, session_success, session_failure },
{ AIRPLAY_SEQ_SEND_VOLUME, session_status, session_failure },
{ AIRPLAY_SEQ_SEND_TEXT, NULL, session_failure },
{ AIRPLAY_SEQ_SEND_PROGRESS, NULL, session_failure },
{ AIRPLAY_SEQ_SEND_ARTWORK, NULL, session_failure },
{ AIRPLAY_SEQ_PAIR_SETUP, session_pair_success, session_failure },
{ AIRPLAY_SEQ_PAIR_VERIFY, session_pair_success, session_failure },
{ AIRPLAY_SEQ_PAIR_TRANSIENT, session_pair_success, session_failure },
{ AIRPLAY_SEQ_FEEDBACK, NULL, session_failure },
};
// The size of the second array dimension MUST at least be the size of largest
// sequence + 1, because then we can count on a zero terminator when iterating
static struct airplay_seq_request airplay_seq_request[][7] =
{
{
// response_handler_options() will determine appropriate sequence to continue with based on device response
{ AIRPLAY_SEQ_START, "OPTIONS", EVRTSP_REQ_OPTIONS, NULL, response_handler_options_start, NULL, "*", true },
},
{
{ AIRPLAY_SEQ_START_RERUN, "OPTIONS (re-run)", EVRTSP_REQ_OPTIONS, NULL, response_handler_options_start, NULL, "*", false },
},
{
// { AIRPLAY_SEQ_START_AP2, "auth-setup", EVRTSP_REQ_POST, payload_make_auth_setup, NULL, "application/octet-stream", "/auth-setup", true },
{ AIRPLAY_SEQ_START_AP2, "SETUP (session)", EVRTSP_REQ_SETUP, payload_make_setup_session, response_handler_setup_session, "application/x-apple-binary-plist", NULL, false },
{ AIRPLAY_SEQ_START_AP2, "SETPEERS", EVRTSP_REQ_SETPEERS, payload_make_setpeers, NULL, "/peer-list-changed", NULL, false },
{ AIRPLAY_SEQ_START_AP2, "SETUP (stream)", EVRTSP_REQ_SETUP, payload_make_setup_stream, response_handler_setup_stream, "application/x-apple-binary-plist", NULL, false },
{ AIRPLAY_SEQ_START_AP2, "SET_PARAMETER (volume)", EVRTSP_REQ_SET_PARAMETER, payload_make_set_volume, response_handler_volume_start, "text/parameters", NULL, true },
{ AIRPLAY_SEQ_START_AP2, "RECORD", EVRTSP_REQ_RECORD, payload_make_record, response_handler_record, NULL, NULL, false },
},
{
{ AIRPLAY_SEQ_PROBE, "OPTIONS (probe)", EVRTSP_REQ_OPTIONS, NULL, response_handler_options_probe, NULL, "*", true },
},
{
{ AIRPLAY_SEQ_FLUSH, "FLUSH", EVRTSP_REQ_FLUSH, payload_make_flush, response_handler_flush, NULL, NULL, false },
},
{
{ AIRPLAY_SEQ_STOP, "TEARDOWN", EVRTSP_REQ_TEARDOWN, payload_make_teardown, response_handler_teardown, NULL, NULL, true },
},
{
{ AIRPLAY_SEQ_FAILURE, "TEARDOWN (failure)", EVRTSP_REQ_TEARDOWN, payload_make_teardown, response_handler_teardown_failure, NULL, NULL, false },
},
{
{ AIRPLAY_SEQ_PIN_START, "PIN start", EVRTSP_REQ_POST, payload_make_pin_start, response_handler_pin_start, NULL, "/pair-pin-start", false },
},
{
{ AIRPLAY_SEQ_SEND_VOLUME, "SET_PARAMETER (volume)", EVRTSP_REQ_SET_PARAMETER, payload_make_set_volume, NULL, "text/parameters", NULL, true },
},
{
{ AIRPLAY_SEQ_SEND_TEXT, "SET_PARAMETER (text)", EVRTSP_REQ_SET_PARAMETER, payload_make_send_text, NULL, "application/x-dmap-tagged", NULL, true },
},
{
{ AIRPLAY_SEQ_SEND_PROGRESS, "SET_PARAMETER (progress)", EVRTSP_REQ_SET_PARAMETER, payload_make_send_progress, NULL, "text/parameters", NULL, true },
},
{
{ AIRPLAY_SEQ_SEND_ARTWORK, "SET_PARAMETER (artwork)", EVRTSP_REQ_SET_PARAMETER, payload_make_send_artwork, NULL, NULL, NULL, true },
},
{
{ AIRPLAY_SEQ_PAIR_SETUP, "pair setup 1", EVRTSP_REQ_POST, payload_make_pair_setup1, response_handler_pair_setup1, "application/octet-stream", "/pair-setup", false },
{ AIRPLAY_SEQ_PAIR_SETUP, "pair setup 2", EVRTSP_REQ_POST, payload_make_pair_setup2, response_handler_pair_setup2, "application/octet-stream", "/pair-setup", false },
{ AIRPLAY_SEQ_PAIR_SETUP, "pair setup 3", EVRTSP_REQ_POST, payload_make_pair_setup3, response_handler_pair_setup3, "application/octet-stream", "/pair-setup", false },
},
{
// Proceed on error is true because we want to delete the device key in the response handler if the verification fails
{ AIRPLAY_SEQ_PAIR_VERIFY, "pair verify 1", EVRTSP_REQ_POST, payload_make_pair_verify1, response_handler_pair_verify1, "application/octet-stream", "/pair-verify", true },
{ AIRPLAY_SEQ_PAIR_VERIFY, "pair verify 2", EVRTSP_REQ_POST, payload_make_pair_verify2, response_handler_pair_verify2, "application/octet-stream", "/pair-verify", false },
},
{
// Some devices (i.e. my ATV4) gives a 470 when trying transient, so we proceed on that so the handler can trigger PIN setup sequence
{ AIRPLAY_SEQ_PAIR_TRANSIENT, "pair setup 1", EVRTSP_REQ_POST, payload_make_pair_setup1, response_handler_pair_setup1, "application/octet-stream", "/pair-setup", true },
{ AIRPLAY_SEQ_PAIR_TRANSIENT, "pair setup 2", EVRTSP_REQ_POST, payload_make_pair_setup2, response_handler_pair_setup2, "application/octet-stream", "/pair-setup", false },
},
{
{ AIRPLAY_SEQ_FEEDBACK, "POST /feedback", EVRTSP_REQ_POST, NULL, NULL, NULL, "/feedback", true },
},
};
static void
sequence_continue_cb(struct evrtsp_request *req, void *arg)
{
struct airplay_seq_ctx *seq_ctx = arg;
struct airplay_seq_request *cur_request = seq_ctx->cur_request;
struct airplay_session *rs = seq_ctx->session;
enum airplay_seq_type seq_type;
rs->reqs_in_flight--;
if (!rs->reqs_in_flight)
evrtsp_connection_set_closecb(rs->ctrl, rtsp_close_cb, rs);
if (!req)
{
DPRINTF(E_LOG, L_RAOP, "No response to %s from '%s'\n", cur_request->name, rs->devname);
goto error;
}
if (req->response_code != RTSP_OK)
{
if (!cur_request->proceed_on_rtsp_not_ok)
{
DPRINTF(E_LOG, L_RAOP, "Response to %s from '%s' was negative, aborting (%d %s)\n", cur_request->name, rs->devname, req->response_code, req->response_code_line);
goto error;
}
DPRINTF(E_WARN, L_RAOP, "Response to %s from '%s' was negative, proceeding anyway (%d %s)\n", cur_request->name, rs->devname, req->response_code, req->response_code_line);
}
// We don't check that the reply CSeq matches the request CSeq, because some
// targets like Reflector and AirFoil don't return the CSeq according to the
// rtsp spec. And the CSeq is not really important anyway.
if (cur_request->response_handler)
{
seq_type = cur_request->response_handler(req, rs);
if (seq_type != AIRPLAY_SEQ_CONTINUE)
{
if (seq_type == AIRPLAY_SEQ_ABORT)
goto error;
// Handler wanted to start a new sequence
sequence_start(seq_type, seq_ctx->session, seq_ctx->payload_make_arg, seq_ctx->log_caller);
free(seq_ctx);
return;
}
}
seq_ctx->cur_request++;
if (seq_ctx->cur_request->name)
{
sequence_continue(seq_ctx);
return;
}
if (seq_ctx->on_success)
seq_ctx->on_success(rs);
free(seq_ctx);
return;
error:
if (seq_ctx->on_error)
seq_ctx->on_error(rs);
free(seq_ctx);
}
static void
sequence_continue(struct airplay_seq_ctx *seq_ctx)
{
struct airplay_session *rs = seq_ctx->session;
struct airplay_seq_request *cur_request = seq_ctx->cur_request;
struct evrtsp_request *req = NULL;
const char *uri;
int ret;
req = evrtsp_request_new(sequence_continue_cb, seq_ctx);
if (!req)
goto error;
ret = request_headers_add(req, rs, cur_request->rtsp_type);
if (ret < 0)
goto error;
if (cur_request->content_type)
evrtsp_add_header(req->output_headers, "Content-Type", cur_request->content_type);
if (cur_request->payload_make)
{
ret = cur_request->payload_make(req, rs, seq_ctx->payload_make_arg);
if (ret > 0) // Skip to next request in sequence, if none -> error
{
seq_ctx->cur_request++;
if (!seq_ctx->cur_request->name)
{
DPRINTF(E_LOG, L_RAOP, "Bug! payload_make signaled skip request, but there is nothing to skip to\n");
goto error;
}
evrtsp_request_free(req);
sequence_continue(seq_ctx);
return;
}
else if (ret < 0)
goto error;
}
uri = (cur_request->uri) ? cur_request->uri : rs->session_url;
DPRINTF(E_DBG, L_RAOP, "%s: Sending %s to '%s'\n", seq_ctx->log_caller, cur_request->name, rs->devname);
ret = evrtsp_make_request(rs->ctrl, req, cur_request->rtsp_type, uri);
if (ret < 0)
goto error;
evrtsp_connection_set_closecb(rs->ctrl, NULL, NULL);
rs->reqs_in_flight++;
return;
error:
DPRINTF(E_LOG, L_RAOP, "%s: Error sending %s to '%s'\n", seq_ctx->log_caller, cur_request->name, rs->devname);
if (req)
evrtsp_request_free(req);
// Sets status to FAILED, gives status to player and frees session. Must be
// deferred, otherwise sequence_start() could invalidate the session, meaning
// any dereference of the session by the caller after sequence_start() would
// segfault.
deferred_session_failure(rs);
free(seq_ctx);
}
// All errors that may occur during a sequence are called back async
static void
sequence_start(enum airplay_seq_type seq_type, struct airplay_session *rs, void *arg, const char *log_caller)
{
struct airplay_seq_ctx *seq_ctx;
CHECK_NULL(L_RAOP, seq_ctx = calloc(1, sizeof(struct airplay_seq_ctx)));
seq_ctx->session = rs;
seq_ctx->cur_request = &airplay_seq_request[seq_type][0]; // First step of the sequence
seq_ctx->on_success = airplay_seq_definition[seq_type].on_success;
seq_ctx->on_error = airplay_seq_definition[seq_type].on_error;
seq_ctx->payload_make_arg = arg;
seq_ctx->log_caller = log_caller;
sequence_continue(seq_ctx); // Ownership transferred
}
/* ---------------- Airplay devices discovery - mDNS callback --------------- */
/* Thread: main (mdns) */
static int
features_parse(struct keyval *features_kv, const char *fs1, const char *fs2, const char *name)
{
uint64_t features;
int i, j;
if (safe_hextou32(fs1, (uint32_t *)&features) < 0 || safe_hextou32(fs2, ((uint32_t *)&features) + 1) < 0)
{
DPRINTF(E_LOG, L_RAOP, "AirPlay '%s': unexpected features field in TXT record!\n", name);
return -1;
}
DPRINTF(E_DBG, L_RAOP, "Parsing features flags from AirPlay '%s': %s (%" PRIu64 ")\n", name, fs1, features);
// Walk through the bits
for (i = 0; i < (sizeof(features) * CHAR_BIT); i++)
{
if (((features >> i) & 0x01) == 0)
continue;
// Check if we have it in the features map
for (j = 0; j < ARRAY_SIZE(features_map); j++)
{
if (i == features_map[j].bit)
{
DPRINTF(E_DBG, L_RAOP, "Speaker '%s' announced feature %d: '%s'\n", name, i, features_map[j].name);
keyval_add(features_kv, features_map[j].name, "1");
break;
}
}
if (j == ARRAY_SIZE(features_map))
DPRINTF(E_DBG, L_RAOP, "Speaker '%s' announced feature %d: 'Unknown'\n", name, i);
}
return 0;
}
/* Examples of txt content:
* Airport Express 2:
["pk=7de...39" "gcgl=0" "gid=0fd...4" "pi=0fd...a4" "srcvers=366.0" "protovers=1.1" "serialNumber=C8...R" "manufacturer=Apple Inc." "model=AirPort10,115" "flags=0x4" "fv=p20.78100.3" "rsf=0x0" "features=0x445D0A00,0x1C340" "deviceid=74:1B:B2:D1:1A:B7" "acl=0"]
* Apple TV 4:
["vv=2" "osvers=14.2" "srcvers=525.38.42" "pk=c4e...c88" "psi=67C...DBC" "pi=b0b...da0" "protovers=1.1" "model=AppleTV5,3" "gcgl=1" "igl=1" "gid=B...73" "flags=0x244" "features=0x5A7FDFD5,0x3C155FDE" "fex=1d9/Wt5fFTw" "deviceid=AA:BB:CC:DD:EE:FF" "btaddr=D0:00:44:66:BB:66" "acl=0"]
* Roku
["pk=xxxxxxxxx” "gcgl=0" "gid=xxxxxxx "psi=xxxxx” "pi=8A:71:CA:EF:xxxx" "srcvers=377.28.01" "protovers=1.1" "serialNumber=xxxxxxx "manufacturer=Roku" "model=3810X" "flags=0x644" "at=0x3" "fv=p20.9.40.4190" "rsf=0x3" "features=0x7F8AD0,0x10BCF46" "deviceid=8A:71:CA:xxxxx” "acl=0"]
* Samsung TV
["pk=7xxxxxxxxxx” "gcgl=0" "gid=xxxxxxxxxxx "psi=xxxxxxx” "pi=4C:6F:64:xxxxxxx "srcvers=377.17.24.6" "protovers=1.1" "serialNumber=xxxxxxx” "manufacturer=Samsung" "model=UNU7090" "flags=0x244" "fv=p20.0.1" "rsf=0x3" "features=0x7F8AD0,0x38BCB46" "deviceid=64:1C:AE:xxxxx "acl=0"]
* HomePod
["vv=2" "osvers=14.3" "srcvers=530.6" "pk=..." "psi=31...D3" "pi=fd...87" "protovers=1.1" "model=AudioAccessory1,1" "tsid=4...E" "gpn=name" "gcgl=1" "igl=1" "gid=4...E" "flags=0x1a404" "features=0x4A7FCA00,0x3C356BD0" "fex=AMp/StBrNTw" "deviceid=D4:...:C1" "btaddr=5E:...:F1" "acl=0"]
*/
static void
airplay_device_cb(const char *name, const char *type, const char *domain, const char *hostname, int family, const char *address, int port, struct keyval *txt)
{
struct output_device *rd;
struct airplay_extra *re;
struct keyval features = { 0 };
cfg_t *devcfg;
cfg_opt_t *cfgopt;
const char *p;
char *s;
char *ptr;
uint64_t id;
int ret;
p = keyval_get(txt, "deviceid");
if (!p)
{
DPRINTF(E_LOG, L_RAOP, "AirPlay device '%s' is missing a device ID\n", name);
return;
}
// Convert AA:BB:CC:DD:EE:FF -> AABBCCDDEEFF -> uint64 id
s = calloc(1, strlen(p) + 1);
for (ptr = s; *p != '\0'; p++)
{
if (*p == ':')
continue;
*ptr = *p;
ptr++;
}
ret = safe_hextou64(s, &id);
free(s);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not extract AirPlay device ID ('%s')\n", name);
return;
}
DPRINTF(E_DBG, L_RAOP, "Event for AirPlay device '%s' (port %d, id %" PRIx64 ")\n", name, port, id);
devcfg = cfg_gettsec(cfg, "airplay", name);
if (devcfg && cfg_getbool(devcfg, "exclude"))
{
DPRINTF(E_LOG, L_RAOP, "Excluding AirPlay device '%s' as set in config\n", name);
return;
}
if (devcfg && cfg_getbool(devcfg, "permanent") && (port < 0))
{
DPRINTF(E_INFO, L_RAOP, "AirPlay device '%s' disappeared, but set as permanent in config\n", name);
return;
}
CHECK_NULL(L_RAOP, rd = calloc(1, sizeof(struct output_device)));
CHECK_NULL(L_RAOP, re = calloc(1, sizeof(struct airplay_extra)));
rd->id = id;
rd->name = strdup(name);
rd->type = OUTPUT_TYPE_AIRPLAY;
rd->type_name = outputs_name(rd->type);
rd->extra_device_info = re;
if (port < 0)
{
// Device stopped advertising
switch (family)
{
case AF_INET:
rd->v4_port = 1;
break;
case AF_INET6:
rd->v6_port = 1;
break;
}
ret = player_device_remove(rd);
if (ret < 0)
goto free_rd;
return;
}
// Features, see features_map[]
p = keyval_get(txt, "features");
if (!p || !strchr(p, ','))
{
DPRINTF(E_LOG, L_RAOP, "AirPlay device '%s' error: Missing/unexpected 'features' in TXT field\n", name);
goto free_rd;
}
ret = features_parse(&features, p, strchr(p, ',') + 1, name);
if (ret < 0)
goto free_rd;
if (!keyval_get(&features, "SupportsAirPlayAudio"))
{
DPRINTF(E_LOG, L_RAOP, "AirPlay device '%s' does not support audio\n", name);
goto free_rd;
}
if (keyval_get(&features, "MetadataFeatures_0"))
re->wanted_metadata |= RAOP_MD_WANTS_ARTWORK;
if (keyval_get(&features, "MetadataFeatures_1"))
re->wanted_metadata |= RAOP_MD_WANTS_PROGRESS;
if (keyval_get(&features, "MetadataFeatures_2"))
re->wanted_metadata |= RAOP_MD_WANTS_TEXT;
if (keyval_get(&features, "Authentication_8"))
re->supports_auth_setup = 1;
if (keyval_get(&features, "SupportsSystemPairing") || keyval_get(&features, "SupportsCoreUtilsPairingAndEncryption"))
re->supports_pairing_transient = 1;
else if (keyval_get(&features, "SupportsHKPairingAndAccessControl"))
rd->requires_auth = 1;
keyval_clear(&features);
// Only default audio quality supported so far
rd->quality.sample_rate = RAOP_QUALITY_SAMPLE_RATE_DEFAULT;
rd->quality.bits_per_sample = RAOP_QUALITY_BITS_PER_SAMPLE_DEFAULT;
rd->quality.channels = RAOP_QUALITY_CHANNELS_DEFAULT;
if (!quality_is_equal(&rd->quality, &airplay_quality_default))
DPRINTF(E_LOG, L_RAOP, "Device '%s' requested non-default audio quality (%d/%d/%d)\n", rd->name, rd->quality.sample_rate, rd->quality.bits_per_sample, rd->quality.channels);
// Device type
re->devtype = RAOP_DEV_OTHER;
p = keyval_get(txt, "model");
if (!p)
re->devtype = RAOP_DEV_APEX1_80211G; // First generation AirPort Express
else if (strncmp(p, "AirPort4", strlen("AirPort4")) == 0)
re->devtype = RAOP_DEV_APEX2_80211N; // Second generation
else if (strncmp(p, "AirPort", strlen("AirPort")) == 0)
re->devtype = RAOP_DEV_APEX3_80211N; // Third generation and newer
else if (strncmp(p, "AppleTV5,3", strlen("AppleTV5,3")) == 0)
re->devtype = RAOP_DEV_APPLETV4; // Stream to ATV with tvOS 10 needs to be kept alive
else if (strncmp(p, "AppleTV", strlen("AppleTV")) == 0)
re->devtype = RAOP_DEV_APPLETV;
else if (strncmp(p, "AudioAccessory", strlen("AudioAccessory")) == 0)
re->devtype = RAOP_DEV_HOMEPOD;
else if (*p == '\0')
DPRINTF(E_LOG, L_RAOP, "AirPlay device '%s': am has no value\n", name);
// If the user didn't set any reconnect setting we enable for Apple TV and
// HomePods due to https://github.com/ejurgensen/forked-daapd/issues/734
cfgopt = devcfg ? cfg_getopt(devcfg, "reconnect") : NULL;
if (cfgopt && cfgopt->nvalues == 1)
rd->resurrect = cfg_opt_getnbool(cfgopt, 0);
else
rd->resurrect = (re->devtype == RAOP_DEV_APPLETV4) || (re->devtype == RAOP_DEV_HOMEPOD);
switch (family)
{
case AF_INET:
rd->v4_address = strdup(address);
rd->v4_port = port;
DPRINTF(E_INFO, L_RAOP, "Adding AirPlay device '%s': password: %u, verification: %u, encrypt: %u, authsetup: %u, metadata: %u, type %s, address %s:%d\n",
name, rd->has_password, rd->requires_auth, re->encrypt, re->supports_auth_setup, re->wanted_metadata, airplay_devtype[re->devtype], address, port);
break;
case AF_INET6:
rd->v6_address = strdup(address);
rd->v6_port = port;
DPRINTF(E_INFO, L_RAOP, "Adding AirPlay device '%s': password: %u, verification: %u, encrypt: %u, authsetup: %u, metadata: %u, type %s, address [%s]:%d\n",
name, rd->has_password, rd->requires_auth, re->encrypt, re->supports_auth_setup, re->wanted_metadata, airplay_devtype[re->devtype], address, port);
break;
default:
DPRINTF(E_LOG, L_RAOP, "Error: AirPlay device '%s' has neither ipv4 og ipv6 address\n", name);
goto free_rd;
}
ret = player_device_add(rd);
if (ret < 0)
goto free_rd;
return;
free_rd:
outputs_device_free(rd);
keyval_clear(&features);
}
/* ---------------------------- Module definitions -------------------------- */
/* Thread: player */
static int
airplay_device_start_generic(struct output_device *device, int callback_id, bool only_probe)
{
struct airplay_session *rs;
rs = session_make(device, callback_id);
if (!rs)
return -1;
// After pairing/device verification, send an OPTIONS request.
if (only_probe)
rs->next_seq = AIRPLAY_SEQ_PROBE;
else
rs->next_seq = AIRPLAY_SEQ_START;
if (device->auth_key)
sequence_start(AIRPLAY_SEQ_PAIR_VERIFY, rs, NULL, "device_start");
else if (rs->pair_type == PAIR_HOMEKIT_TRANSIENT)
sequence_start(AIRPLAY_SEQ_PAIR_TRANSIENT, rs, NULL, "device_start");
else
sequence_start(AIRPLAY_SEQ_PIN_START, rs, NULL, "device_start");
return 1;
}
static int
airplay_device_probe(struct output_device *device, int callback_id)
{
return airplay_device_start_generic(device, callback_id, 1);
}
static int
airplay_device_start(struct output_device *device, int callback_id)
{
return airplay_device_start_generic(device, callback_id, 0);
}
static int
airplay_device_stop(struct output_device *device, int callback_id)
{
struct airplay_session *rs = device->session;
rs->callback_id = callback_id;
sequence_start(AIRPLAY_SEQ_STOP, rs, NULL, "device_stop");
return 1;
}
static int
airplay_device_flush(struct output_device *device, int callback_id)
{
struct airplay_session *rs = device->session;
if (rs->state != AIRPLAY_STATE_STREAMING)
return 0; // No-op, nothing to flush
rs->callback_id = callback_id;
sequence_start(AIRPLAY_SEQ_FLUSH, rs, NULL, "flush");
return 1;
}
static void
airplay_device_cb_set(struct output_device *device, int callback_id)
{
struct airplay_session *rs = device->session;
rs->callback_id = callback_id;
}
static void
airplay_device_free_extra(struct output_device *device)
{
struct airplay_extra *re = device->extra_device_info;
free(re);
}
static int
airplay_device_authorize(struct output_device *device, const char *pin, int callback_id)
{
struct airplay_session *rs;
// Make a session so we can communicate with the device
rs = session_make(device, callback_id);
if (!rs)
return -1;
sequence_start(AIRPLAY_SEQ_PAIR_SETUP, rs, (void *)pin, "device_authorize");
return 1;
}
static void
airplay_write(struct output_buffer *obuf)
{
struct airplay_master_session *rms;
struct airplay_session *rs;
int i;
for (rms = airplay_master_sessions; rms; rms = rms->next)
{
for (i = 0; obuf->data[i].buffer; i++)
{
if (!quality_is_equal(&obuf->data[i].quality, &rms->rtp_session->quality))
continue;
// Set rms->cur_stamp, which involves a calculation of which session
// rtptime corresponds to the pts we are given by the player.
timestamp_set(rms, obuf->pts);
// Sends sync packets to new sessions, and if it is sync time then also to old sessions
packets_sync_send(rms);
// TODO avoid this copy
evbuffer_add(rms->evbuf, obuf->data[i].buffer, obuf->data[i].bufsize);
rms->evbuf_samples += obuf->data[i].samples;
// Send as many packets as we have data for (one packet requires rawbuf_size bytes)
while (evbuffer_get_length(rms->evbuf) >= rms->rawbuf_size)
{
evbuffer_remove(rms->evbuf, rms->rawbuf, rms->rawbuf_size);
rms->evbuf_samples -= rms->samples_per_packet;
packets_send(rms);
}
}
}
// Check for devices that have joined since last write (we have already sent them
// initialization sync and rtp packets via packets_sync_send and packets_send)
for (rs = airplay_sessions; rs; rs = rs->next)
{
if (rs->state != AIRPLAY_STATE_CONNECTED)
continue;
// Start sending progress to keep ATV's alive
if (!event_pending(keep_alive_timer, EV_TIMEOUT, NULL))
evtimer_add(keep_alive_timer, &keep_alive_tv);
rs->state = AIRPLAY_STATE_STREAMING;
// Make a cb?
}
}
static int
airplay_init(void)
{
int v6enabled;
int family;
int ret;
int i;
timing_4svc.fd = -1;
timing_4svc.port = 0;
timing_6svc.fd = -1;
timing_6svc.port = 0;
control_4svc.fd = -1;
control_4svc.port = 0;
control_6svc.fd = -1;
control_6svc.port = 0;
// Check alignment of enum seq_type with airplay_seq_definition and
// airplay_seq_request
for (i = 0; i < ARRAY_SIZE(airplay_seq_definition); i++)
{
if (airplay_seq_definition[i].seq_type != i || airplay_seq_request[i][0].seq_type != i)
{
DPRINTF(E_LOG, L_RAOP, "Bug! Misalignment between sequence enum and structs: %d, %d, %d\n", i, airplay_seq_definition[i].seq_type, airplay_seq_request[i][0].seq_type);
return -1;
}
}
CHECK_NULL(L_RAOP, keep_alive_timer = evtimer_new(evbase_player, airplay_keep_alive_timer_cb, NULL));
v6enabled = cfg_getbool(cfg_getsec(cfg, "general"), "ipv6");
ret = airplay_timing_start(v6enabled);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "AirPlay time synchronization failed to start\n");
goto out_free_timer;
}
ret = airplay_control_start(v6enabled);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "AirPlay playback control failed to start\n");
goto out_stop_timing;
}
if (v6enabled)
v6enabled = !((timing_6svc.fd < 0) || (control_6svc.fd < 0));
if (v6enabled)
family = AF_UNSPEC;
else
family = AF_INET;
ret = mdns_browse("_airplay._tcp", family, airplay_device_cb, MDNS_CONNECTION_TEST);
if (ret < 0)
{
DPRINTF(E_LOG, L_RAOP, "Could not add mDNS browser for AirPlay devices\n");
goto out_stop_control;
}
return 0;
out_stop_control:
airplay_control_stop();
out_stop_timing:
airplay_timing_stop();
out_free_timer:
event_free(keep_alive_timer);
return -1;
}
static void
airplay_deinit(void)
{
struct airplay_session *rs;
for (rs = airplay_sessions; airplay_sessions; rs = airplay_sessions)
{
airplay_sessions = rs->next;
session_free(rs);
}
airplay_control_stop();
airplay_timing_stop();
event_free(keep_alive_timer);
}
struct output_definition output_airplay =
{
.name = "AirPlay 2",
.type = OUTPUT_TYPE_AIRPLAY,
.priority = 1,
.disabled = 0,
.init = airplay_init,
.deinit = airplay_deinit,
.device_start = airplay_device_start,
.device_stop = airplay_device_stop,
.device_flush = airplay_device_flush,
.device_probe = airplay_device_probe,
.device_cb_set = airplay_device_cb_set,
.device_free_extra = airplay_device_free_extra,
.device_volume_set = airplay_set_volume_one,
.device_volume_to_pct = airplay_volume_to_pct,
.write = airplay_write,
.metadata_prepare = airplay_metadata_prepare,
.metadata_send = airplay_metadata_send,
.metadata_purge = airplay_metadata_purge,
.device_authorize = airplay_device_authorize,
};