+ return 1;
+}
+
+static void recv_afs_result(void)
+{
+ int ret, passed_fd = -1, shmid;
+ uint32_t afs_code = 0, afs_data = 0;
+ struct stat statbuf;
+ struct timeval now;
+
+ PARA_NOTICE_LOG("recv\n");
+ ret = recv_afs_msg(&passed_fd, &afs_code, &afs_data);
+ if (ret < 0)
+ goto err;
+ PARA_NOTICE_LOG("got the fd: %d, code: %u, shmid: %u\n",
+ passed_fd, afs_code, afs_data);
+ ret = -E_BAD_AFS_CODE;
+ if (afs_code != NEXT_AUDIO_FILE)
+ goto err;
+ afsss = AFS_SOCKET_READY;
+ shmid = afs_data;
+ ret = load_afd(shmid, &mmd->afd);
+ if (ret < 0)
+ goto err;
+ shm_destroy(shmid);
+ PARA_NOTICE_LOG("next audio file: %s (%lu chunks)\n", mmd->afd.path,
+ mmd->afd.afhi.chunks_total);
+ ret = fstat(passed_fd, &statbuf);
+ if (ret < 0) {
+ PARA_ERROR_LOG("fstat error:\n");
+ ret = -ERRNO_TO_PARA_ERROR(errno);
+ goto err;
+ }
+ mmd->size = statbuf.st_size;
+ mmd->mtime = statbuf.st_mtime;
+ map = para_mmap(mmd->size, PROT_READ, MAP_PRIVATE,
+ passed_fd, 0);
+ close(passed_fd);
+ strcpy(mmd->filename, mmd->afd.path); /* FIXME: check length */
+ mmd->afi.header_len = 0; /* default: no header */
+ mmd->audio_format = mmd->afd.afsi.audio_format_id;
+ mmd->chunks_sent = 0;
+ mmd->current_chunk = 0;
+ mmd->offset = 0;
+ mmd->events++;
+ mmd->num_played++;
+ mmd->afi = mmd->afd.afhi;
+ mmd->new_vss_status_flags &= (~VSS_NEXT);
+ gettimeofday(&now, NULL);
+ tv_add(&now, &announce_tv, &data_send_barrier);
+ return;
+err:
+ if (passed_fd >= 0)
+ close(passed_fd);
+ PARA_ERROR_LOG("%s\n", PARA_STRERROR(-ret));
+}
+
+void vss_post_select(fd_set *rfds, fd_set *wfds)
+{
+ int ret;
+
+ if (FD_ISSET(afs_socket, rfds))
+ recv_afs_result();
+ if (afsss != AFS_SOCKET_CHECK_FOR_WRITE || !FD_ISSET(afs_socket, wfds))
+ return;
+ PARA_NOTICE_LOG("requesting new socket\n");
+ ret = send_buffer(afs_socket, "new");
+ afsss = AFS_SOCKET_AFD_PENDING;
+}
+
+static void get_chunk(long unsigned chunk_num, char **buf, size_t *len)
+{
+ size_t pos = mmd->afi.chunk_table[chunk_num];
+ *buf = map + pos;
+ *len = mmd->afi.chunk_table[chunk_num + 1] - pos;
+
+ if (chunk_num + 5 > mmd->afd.afhi.chunks_total)
+ PARA_NOTICE_LOG("chunk %lu/%lu\n, len: %zu\n", chunk_num,
+ mmd->afd.afhi.chunks_total, *len);
+}
+
+/**
+ * Get the data of the given chunk.
+ *
+ * \param chunk_num The number of the desired chunk.
+ * \param buf Chunk data.
+ * \param len Chunk length in bytes.
+ *
+ * \return Positive on success, negative on errors.
+ */
+int vss_get_chunk(long unsigned chunk_num, char **buf, size_t *len)
+{
+ if (mmd->audio_format < 0 || !map || !vss_playing())
+ return -E_CHUNK;
+ if (chunk_num >= mmd->afi.chunks_total)
+ return -E_CHUNK;
+ get_chunk(chunk_num, buf, len);
+ return 1;