Re: [RFCv0 07/14] android/hal-sco: Use global sco file descriptor

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Andrei,

On Thu, May 22, 2014 at 3:06 PM, Andrei Emeltchenko
<Andrei.Emeltchenko.news@xxxxxxxxx> wrote:
> From: Andrei Emeltchenko <andrei.emeltchenko@xxxxxxxxx>
>
> 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@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html



-- 
Luiz Augusto von Dentz
--
To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Bluez Devel]     [Linux Wireless Networking]     [Linux Wireless Personal Area Networking]     [Linux ATH6KL]     [Linux USB Devel]     [Linux Media Drivers]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Big List of Linux Books]

  Powered by Linux