From: Luiz Augusto von Dentz <luiz.dentz@gmail.com>
To: Andrei Emeltchenko <Andrei.Emeltchenko.news@gmail.com>
Cc: "linux-bluetooth@vger.kernel.org" <linux-bluetooth@vger.kernel.org>
Subject: Re: [RFCv0 07/14] android/hal-sco: Use global sco file descriptor
Date: Tue, 27 May 2014 16:08:09 +0300 [thread overview]
Message-ID: <CABBYNZLMKcj_fpSNfpHftuzWiWA=WTOotseftrBf2202iqT-PA@mail.gmail.com> (raw)
In-Reply-To: <1400760367-24915-7-git-send-email-Andrei.Emeltchenko.news@gmail.com>
Hi Andrei,
On Thu, May 22, 2014 at 3:06 PM, Andrei Emeltchenko
<Andrei.Emeltchenko.news@gmail.com> wrote:
> From: Andrei Emeltchenko <andrei.emeltchenko@intel.com>
>
> Android may open input/output stream independently so we use global sco
> file descriptor and mutexes.
> ---
> android/hal-sco.c | 85 ++++++++++++++++++++++++++++++++-----------------------
> 1 file changed, 50 insertions(+), 35 deletions(-)
>
> diff --git a/android/hal-sco.c b/android/hal-sco.c
> index e940547..8c2549b 100644
> --- a/android/hal-sco.c
> +++ b/android/hal-sco.c
> @@ -46,6 +46,10 @@
> static int listen_sk = -1;
> static int ipc_sk = -1;
>
> +static int sco_fd = -1;
> +static uint16_t sco_mtu = 0;
> +static pthread_mutex_t sco_mutex = PTHREAD_MUTEX_INITIALIZER;
> +
Do we really need a mutex for this? I mean this is almost the same as
protecting the device itself for opening concurrent input and output,
if that is the case then I think it is better to put the mutex there.
> static pthread_t ipc_th = 0;
> static pthread_mutex_t sk_mutex = PTHREAD_MUTEX_INITIALIZER;
>
> @@ -53,7 +57,6 @@ struct sco_audio_config {
> uint32_t rate;
> uint32_t channels;
> uint32_t frame_num;
> - uint16_t mtu;
> audio_format_t format;
> };
>
> @@ -61,7 +64,6 @@ struct sco_stream_out {
> struct audio_stream_out stream;
>
> struct sco_audio_config cfg;
> - int fd;
>
> uint8_t *downmix_buf;
> uint8_t *cache;
> @@ -79,7 +81,6 @@ struct sco_stream_in {
> struct audio_stream_in stream;
>
> struct sco_audio_config cfg;
> - int fd;
> };
>
> struct sco_dev {
> @@ -257,18 +258,25 @@ failed:
> return SCO_STATUS_FAILED;
> }
>
> -static int ipc_connect_sco(int *fd, uint16_t *mtu)
> +static int ipc_connect_sco(void)
> {
> struct sco_rsp_connect rsp;
> size_t rsp_len = sizeof(rsp);
> - int ret;
> + int ret = SCO_STATUS_SUCCESS;
>
> DBG("");
>
> - ret = sco_ipc_cmd(SCO_SERVICE_ID, SCO_OP_CONNECT, 0, NULL, &rsp_len,
> - &rsp, fd);
> + pthread_mutex_lock(&sco_mutex);
> +
> + if (sco_fd < 0) {
> + ret = sco_ipc_cmd(SCO_SERVICE_ID, SCO_OP_CONNECT, 0, NULL,
> + &rsp_len, &rsp, &sco_fd);
> +
> + /* Sometimes mtu returned is wrong */
> + sco_mtu = /* rsp.mtu */ 48;
> + }
>
> - *mtu = rsp.mtu;
> + pthread_mutex_unlock(&sco_mutex);
>
> return ret;
> }
> @@ -311,28 +319,27 @@ static bool write_data(struct sco_stream_out *out, const uint8_t *buffer,
> struct pollfd pfd;
> size_t len, written = 0;
> int ret;
> - uint16_t mtu = out->cfg.mtu;
> uint8_t *p;
> uint64_t audio_sent_us, audio_passed_us;
>
> - pfd.fd = out->fd;
> - pfd.events = POLLOUT | POLLIN | POLLHUP | POLLNVAL;
> + pfd.fd = sco_fd;
> + pfd.events = POLLOUT | POLLHUP | POLLNVAL;
>
> while (bytes > written) {
> struct timespec now;
>
> /* poll for sending */
> if (poll(&pfd, 1, SOCKET_POLL_TIMEOUT_MS) == 0) {
> - DBG("timeout fd %d", out->fd);
> + DBG("timeout fd %d", sco_fd);
> return false;
> }
>
> if (pfd.revents & (POLLHUP | POLLNVAL)) {
> - error("error fd %d, events 0x%x", out->fd, pfd.revents);
> + error("error fd %d, events 0x%x", sco_fd, pfd.revents);
> return false;
> }
>
> - len = bytes - written > mtu ? mtu : bytes - written;
> + len = bytes - written > sco_mtu ? sco_mtu : bytes - written;
>
> clock_gettime(CLOCK_REALTIME, &now);
> /* Mark start of the stream */
> @@ -357,11 +364,11 @@ static bool write_data(struct sco_stream_out *out, const uint8_t *buffer,
> if (out->cache_len) {
> DBG("First packet cache_len %zd", out->cache_len);
> memcpy(out->cache + out->cache_len, buffer,
> - mtu - out->cache_len);
> + sco_mtu - out->cache_len);
> p = out->cache;
> }
>
> - if (bytes - written >= mtu)
> + if (bytes - written >= sco_mtu)
> p = (void *) buffer + written;
> else {
> memcpy(out->cache, buffer + written, bytes - written);
> @@ -371,10 +378,10 @@ static bool write_data(struct sco_stream_out *out, const uint8_t *buffer,
> continue;
> }
>
> - ret = write(out->fd, p, len);
> + ret = write(sco_fd, p, len);
> if (ret > 0) {
> if (out->cache_len) {
> - written = mtu - out->cache_len;
> + written = sco_mtu - out->cache_len;
> out->cache_len = 0;
> } else
> written += ret;
> @@ -394,7 +401,7 @@ static bool write_data(struct sco_stream_out *out, const uint8_t *buffer,
>
> if (errno != EINTR) {
> ret = errno;
> - error("write failed (%d) fd %d bytes %zd", ret, out->fd,
> + error("write failed (%d) fd %d bytes %zd", ret, sco_fd,
> bytes);
> return false;
> }
> @@ -414,7 +421,7 @@ static ssize_t out_write(struct audio_stream_out *stream, const void *buffer,
> void *send_buf = out->downmix_buf;
> size_t total;
>
> - DBG("write to fd %d bytes %zu", out->fd, bytes);
> + DBG("write to fd %d bytes %zu", sco_fd, bytes);
>
> if (!out->downmix_buf) {
> error("sco: downmix buffer not initialized");
> @@ -585,19 +592,17 @@ static int sco_open_output_stream(struct audio_hw_device *dev,
> {
> struct sco_dev *adev = (struct sco_dev *) dev;
> struct sco_stream_out *out;
> - int fd = -1;
> int chan_num, ret;
> size_t resample_size;
> - uint16_t mtu;
>
> DBG("config %p device flags 0x%02x", config, devices);
>
> - if (ipc_connect_sco(&fd, &mtu) != SCO_STATUS_SUCCESS) {
> + if (ipc_connect_sco() != SCO_STATUS_SUCCESS) {
> error("sco: cannot get fd");
> return -EIO;
> }
>
> - DBG("got sco fd %d mtu %u", fd, mtu);
> + DBG("got sco fd %d mtu %u", sco_fd, sco_mtu);
>
> out = calloc(1, sizeof(struct sco_stream_out));
> if (!out)
> @@ -636,16 +641,13 @@ static int sco_open_output_stream(struct audio_hw_device *dev,
>
> out->cfg.frame_num = OUT_STREAM_FRAMES;
>
> - /* we get wrong mtu size for some reason */
> - out->cfg.mtu = /* mtu */ 48;
> -
> out->downmix_buf = malloc(out_get_buffer_size(&out->stream.common));
> if (!out->downmix_buf) {
> free(out);
> return -ENOMEM;
> }
>
> - out->cache = malloc(out->cfg.mtu);
> + out->cache = malloc(sco_mtu);
> if (!out->cache) {
> free(out->downmix_buf);
> free(out);
> @@ -691,7 +693,6 @@ static int sco_open_output_stream(struct audio_hw_device *dev,
>
> *stream_out = &out->stream;
> adev->out = out;
> - out->fd = fd;
>
> return 0;
> failed:
> @@ -704,18 +705,30 @@ failed:
> return ret;
> }
>
> +static void close_sco_socket(void)
> +{
> + DBG("");
> +
> + pthread_mutex_lock(&sco_mutex);
> +
> + if (sco_fd >= 0) {
> + shutdown(sco_fd, SHUT_RDWR);
> + close(sco_fd);
> + sco_fd = -1;
> + }
> +
> + pthread_mutex_unlock(&sco_mutex);
> +}
> +
> static void sco_close_output_stream(struct audio_hw_device *dev,
> struct audio_stream_out *stream_out)
> {
> struct sco_dev *sco_dev = (struct sco_dev *) dev;
> struct sco_stream_out *out = (struct sco_stream_out *) stream_out;
>
> - DBG("dev %p stream %p fd %d", dev, out, sco_dev->out->fd);
> + DBG("dev %p stream %p fd %d", dev, out, sco_fd);
>
> - if (out && out->fd >= 0) {
> - close(out->fd);
> - out->fd = -1;
> - }
> + close_sco_socket();
>
> free(out->cache);
> free(out->downmix_buf);
> @@ -965,7 +978,9 @@ static void sco_close_input_stream(struct audio_hw_device *dev,
> struct sco_dev *sco_dev = (struct sco_dev *) dev;
> struct sco_stream_in *in = (struct sco_stream_in *) stream_in;
>
> - DBG("dev %p stream %p fd %d", dev, in, in->fd);
> + DBG("dev %p stream %p fd %d", dev, in, sco_fd);
> +
> + close_sco_socket();
>
> free(in);
> sco_dev->in = NULL;
> --
> 1.8.3.2
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at http://vger.kernel.org/majordomo-info.html
--
Luiz Augusto von Dentz
next prev parent reply other threads:[~2014-05-27 13:08 UTC|newest]
Thread overview: 19+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-05-22 12:05 [RFCv0 01/14] android/hal-sco: Use nanosleep for SCO synchronization Andrei Emeltchenko
2014-05-22 12:05 ` [RFCv0 02/14] android/hal-sco: Fixes for unreliable mtu Andrei Emeltchenko
2014-05-22 12:05 ` [RFCv0 03/14] android/hal-sco: Add SCO packet cache Andrei Emeltchenko
2014-05-22 12:05 ` [RFCv0 04/14] android/hal-sco: Make use of config parameter Andrei Emeltchenko
2014-05-22 12:05 ` [RFCv0 05/14] android/hal-sco: Implement open input stream Andrei Emeltchenko
2014-05-22 12:05 ` [RFCv0 06/14] android/hal-sco: Check file descriptor >= 0 Andrei Emeltchenko
2014-05-22 12:06 ` [RFCv0 07/14] android/hal-sco: Use global sco file descriptor Andrei Emeltchenko
2014-05-27 13:08 ` Luiz Augusto von Dentz [this message]
2014-05-28 10:24 ` Andrei Emeltchenko
2014-05-28 10:32 ` Luiz Augusto von Dentz
2014-05-22 12:06 ` [RFCv0 08/14] android/haltest: Add open/close input stream commands Andrei Emeltchenko
2014-05-22 12:06 ` [RFCv0 09/14] android/haltest: Add read command Andrei Emeltchenko
2014-05-22 12:06 ` [RFCv0 10/14] android/haltest: Add loop command Andrei Emeltchenko
2014-05-22 12:06 ` [RFCv0 11/14] android/hal-sco: Make debug more readable Andrei Emeltchenko
2014-05-22 12:06 ` [RFCv0 12/14] android/hal-sco: Fix memory leak Andrei Emeltchenko
2014-05-27 13:40 ` Luiz Augusto von Dentz
2014-05-22 12:06 ` [RFCv0 13/14] android/hal-sco: Implement read Andrei Emeltchenko
2014-05-22 12:06 ` [RFCv0 14/14] android/haltest: Implement read to file Andrei Emeltchenko
2014-05-27 13:26 ` [RFCv0 01/14] android/hal-sco: Use nanosleep for SCO synchronization Luiz Augusto von Dentz
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to='CABBYNZLMKcj_fpSNfpHftuzWiWA=WTOotseftrBf2202iqT-PA@mail.gmail.com' \
--to=luiz.dentz@gmail.com \
--cc=Andrei.Emeltchenko.news@gmail.com \
--cc=linux-bluetooth@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).