/* Copyright (c) 2014 The Chromium OS Authors. All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include <stdlib.h> #include <sys/param.h> #include "cras_types.h" #include "buffer_share.h" static inline struct id_offset *find_unused(const struct buffer_share *mix) { unsigned int i; for (i = 0; i < mix->id_sz; i++) { if (!mix->wr_idx[i].used) return &mix->wr_idx[i]; } return NULL; } static inline struct id_offset *find_id(const struct buffer_share *mix, unsigned int id) { unsigned int i; for (i = 0; i < mix->id_sz; i++) { if (mix->wr_idx[i].used && id == mix->wr_idx[i].id) return &mix->wr_idx[i]; } return NULL; } static void alloc_more_ids(struct buffer_share *mix) { unsigned int new_size = mix->id_sz * 2; unsigned int i; mix->wr_idx = realloc(mix->wr_idx, sizeof(mix->wr_idx[0]) * new_size); for (i = mix->id_sz; i < new_size; i++) mix->wr_idx[i].used = 0; mix->id_sz = new_size; } struct buffer_share *buffer_share_create(unsigned int buf_sz) { struct buffer_share *mix; mix = calloc(1, sizeof(*mix)); mix->id_sz = INITIAL_ID_SIZE; mix->wr_idx = calloc(mix->id_sz, sizeof(mix->wr_idx[0])); mix->buf_sz = buf_sz; return mix; } void buffer_share_destroy(struct buffer_share *mix) { if (!mix) return; free(mix->wr_idx); free(mix); } int buffer_share_add_id(struct buffer_share *mix, unsigned int id, void *data) { struct id_offset *o; o = find_id(mix, id); if (o) return -EEXIST; o = find_unused(mix); if (!o) alloc_more_ids(mix); o = find_unused(mix); o->used = 1; o->id = id; o->offset = 0; o->data = data; return 0; } int buffer_share_rm_id(struct buffer_share *mix, unsigned int id) { struct id_offset *o; o = find_id(mix, id); if (!o) return -ENOENT; o->used = 0; o->data = NULL; return 0; } int buffer_share_offset_update(struct buffer_share *mix, unsigned int id, unsigned int delta) { unsigned int i; for (i = 0; i < mix->id_sz; i++) { if (id != mix->wr_idx[i].id) continue; mix->wr_idx[i].offset += delta; break; } return 0; } unsigned int buffer_share_get_new_write_point(struct buffer_share *mix) { unsigned int min_written = mix->buf_sz + 1; unsigned int i; for (i = 0; i < mix->id_sz; i++) { struct id_offset *o = &mix->wr_idx[i]; if (!o->used) continue; min_written = MIN(min_written, o->offset); } for (i = 0; i < mix->id_sz; i++) { struct id_offset *o = &mix->wr_idx[i]; o->offset -= min_written; } if (min_written > mix->buf_sz) return 0; return min_written; } static struct id_offset *get_id_offset(const struct buffer_share *mix, unsigned int id) { unsigned int i; struct id_offset *o; for (i = 0; i < mix->id_sz; i++) { o = &mix->wr_idx[i]; if (o->used && o->id == id) return o; } return NULL; } unsigned int buffer_share_id_offset(const struct buffer_share *mix, unsigned int id) { struct id_offset *o = get_id_offset(mix, id); return o ? o->offset : 0; } void *buffer_share_get_data(const struct buffer_share *mix, unsigned int id) { struct id_offset *o = get_id_offset(mix, id); return o ? o->data : NULL; }