📄 solaris_audio.c
字号:
/* Ogle - A video player * Copyright (C) 2002 Bj鰎n Englund * * 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 LIBOGLEAO_SOLARIS#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <inttypes.h>#include <sys/ioctl.h>#include <fcntl.h>#include <sys/audioio.h>#include <sys/types.h>#ifdef HAVE_STROPTS_H#include <stropts.h>#endif#include <sys/conf.h>#include "ogle_ao.h"#include "ogle_ao_private.h"typedef struct solaris_instance_s { ogle_ao_instance_t ao; int fd; int sample_rate; int samples_written; int sample_frame_size; int initialized;} solaris_instance_t;staticint solaris_init(ogle_ao_instance_t *_instance, ogle_ao_audio_info_t *audio_info){ int single_sample_size; audio_info_t info; solaris_instance_t *instance = (solaris_instance_t *)_instance; if(instance->initialized) { audio_info_t info; ioctl(instance->fd, AUDIO_DRAIN, 0); AUDIO_INITINFO(&info); info.play.error = 0; info.play.samples = 0; if(ioctl(instance->fd, AUDIO_SETINFO, &info) == -1) { perror("AUDIO_SETINFO"); } instance->initialized = 0; instance->sample_rate = 0; instance->samples_written = 0; instance->sample_frame_size = 0; //return -1; // for now we must close and open the device for reinit } AUDIO_INITINFO(&info); info.play.sample_rate = audio_info->sample_rate; info.play.precision = audio_info->sample_resolution; if(audio_info->chtypes != OGLE_AO_CHTYPE_UNSPECIFIED) { // do this only if we have requested specific channels in chtypes // otherwise trust the nr channels audio_info->chtypes = OGLE_AO_CHTYPE_LEFT | OGLE_AO_CHTYPE_RIGHT; audio_info->channels = 2; } if(audio_info->chlist) { free(audio_info->chlist); } audio_info->chlist = NULL; info.play.channels = audio_info->channels; switch(audio_info->encoding) { case OGLE_AO_ENCODING_LINEAR: default: info.play.encoding = AUDIO_ENCODING_LINEAR; break; } /* set the audio format */ if(ioctl(instance->fd, AUDIO_SETINFO, &info) == -1) { perror("setinfo"); return -1; } /* Check that we actually got the requested number of channels, * the frequency and precision that we asked for? */ if(audio_info->chtypes != OGLE_AO_CHTYPE_UNSPECIFIED) { audio_info->chlist = malloc(info.play.channels * sizeof(ogle_ao_chtype_t)); audio_info->chlist[0] = OGLE_AO_CHTYPE_LEFT; audio_info->chlist[1] = OGLE_AO_CHTYPE_RIGHT; } /* Stored as 8bit, 16bit, or 32bit */ single_sample_size = (info.play.precision + 7) / 8; if(single_sample_size > 2) { single_sample_size = 4; } instance->sample_frame_size = single_sample_size*info.play.channels; audio_info->sample_frame_size = instance->sample_frame_size; instance->initialized = 1; return 0;}staticint solaris_play(ogle_ao_instance_t *_instance, void *samples, size_t nbyte){ solaris_instance_t *instance = (solaris_instance_t *)_instance; int written; written = write(instance->fd, samples, nbyte); if(written == -1) { perror("audio write"); return -1; } instance->samples_written += written / instance->sample_frame_size; return 0;}staticint solaris_odelay(ogle_ao_instance_t *_instance, uint32_t *samples_return){ solaris_instance_t *instance = (solaris_instance_t *)_instance; audio_info_t info; int res; int odelay; static int e = 0; unsigned int samples_played; res = ioctl(instance->fd, AUDIO_GETINFO, &info); if(res == -1) { perror("AUDIO_GETINFO"); return -1; } samples_played = info.play.samples; /* fprintf(stderr, ":: %d :: %d -- e: %d, %d\n", instance->samples_written, info.play.samples, info.play.error, e); */ if(info.play.error) { e++; AUDIO_INITINFO(&info); info.play.error = 0; info.play.samples = 0; res = ioctl(instance->fd, AUDIO_SETINFO, &info); if(res == -1) { perror("AUDIO_SETINFO"); } if(instance->samples_written != samples_played) { fprintf(stderr, "underrun and more samples played: %d, %d\n", samples_played, instance->samples_written); } info.play.samples = 0; instance->samples_written = 0; } if(instance->samples_written < info.play.samples) { //something is wrong with the accounting fprintf(stderr, "written: %d < played: %d\n", instance->samples_written, info.play.samples); } odelay = instance->samples_written - info.play.samples; // fprintf(stderr, "odelay: %d\n", odelay); *samples_return = odelay; return 0;}staticvoid solaris_close(ogle_ao_instance_t *_instance){ solaris_instance_t *instance = (solaris_instance_t *)_instance; close(instance->fd);}staticint solaris_flush(ogle_ao_instance_t *_instance){ solaris_instance_t *instance = (solaris_instance_t *)_instance; audio_info_t info; AUDIO_INITINFO(&info); info.play.pause = 1; ioctl(instance->fd, AUDIO_SETINFO, &info); ioctl(instance->fd, I_FLUSH, FLUSHW); AUDIO_INITINFO(&info); info.play.pause = 0; info.play.error = 0; info.play.samples = 0; ioctl(instance->fd, AUDIO_SETINFO, &info); instance->samples_written = 0; ioctl(instance->fd, AUDIO_DRAIN, 0); return 0;}staticint solaris_drain(ogle_ao_instance_t *_instance){ solaris_instance_t *instance = (solaris_instance_t *)_instance; ioctl(instance->fd, AUDIO_DRAIN, 0); return 0;}static ogle_ao_instance_t *solaris_open(char *dev){ solaris_instance_t *instance; instance = malloc(sizeof(solaris_instance_t)); if(instance == NULL) return NULL; instance->ao.init = solaris_init; instance->ao.play = solaris_play; instance->ao.close = solaris_close; instance->ao.odelay = solaris_odelay; instance->ao.flush = solaris_flush; instance->ao.drain = solaris_drain; instance->initialized = 0; instance->sample_rate = 0; instance->samples_written = 0; instance->sample_frame_size = 0; instance->fd = open(dev, O_WRONLY); if(instance->fd < 0) { fprintf(stderr, "Can not open %s\n", dev); free(instance); return NULL; } return (ogle_ao_instance_t *)instance;}/* The one directly exported function */ogle_ao_instance_t *ao_solaris_open(char *dev){ return solaris_open(dev);}#endif
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -