Home | History | Annotate | Line # | Download | only in dev
      1  1.3  tsutsui /*	$NetBSD: psgpam_enc.c,v 1.3 2023/01/15 05:08:33 tsutsui Exp $	*/
      2  1.1  tsutsui 
      3  1.1  tsutsui /*
      4  1.1  tsutsui  * Copyright (c) 2018 Yosuke Sugahara. All rights reserved.
      5  1.1  tsutsui  *
      6  1.1  tsutsui  * Redistribution and use in source and binary forms, with or without
      7  1.1  tsutsui  * modification, are permitted provided that the following conditions
      8  1.1  tsutsui  * are met:
      9  1.1  tsutsui  * 1. Redistributions of source code must retain the above copyright
     10  1.1  tsutsui  *    notice, this list of conditions and the following disclaimer.
     11  1.1  tsutsui  * 2. Redistributions in binary form must reproduce the above copyright
     12  1.1  tsutsui  *    notice, this list of conditions and the following disclaimer in the
     13  1.1  tsutsui  *    documentation and/or other materials provided with the distribution.
     14  1.1  tsutsui  *
     15  1.1  tsutsui  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     16  1.1  tsutsui  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     17  1.1  tsutsui  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     18  1.1  tsutsui  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
     19  1.1  tsutsui  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
     20  1.1  tsutsui  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
     21  1.1  tsutsui  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
     22  1.1  tsutsui  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
     23  1.1  tsutsui  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     24  1.1  tsutsui  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     25  1.1  tsutsui  * SUCH DAMAGE.
     26  1.1  tsutsui  */
     27  1.1  tsutsui 
     28  1.1  tsutsui /*
     29  1.1  tsutsui  * PSGPAM, PSGPCM encoders
     30  1.1  tsutsui  * Function names are PSGPAM, but also used by PSGPCM.
     31  1.1  tsutsui  *
     32  1.1  tsutsui  * PSGPAM and PSGPCM internally use the unsigned type auint_t for
     33  1.1  tsutsui  * intermediate calculations to manage non-linear conversions.
     34  1.1  tsutsui  */
     35  1.1  tsutsui 
     36  1.1  tsutsui #include <sys/types.h>
     37  1.1  tsutsui 
     38  1.1  tsutsui #if defined(_KERNEL)
     39  1.1  tsutsui #include <sys/device.h>
     40  1.1  tsutsui #include <sys/audioio.h>
     41  1.1  tsutsui #include <dev/audio/audio_if.h>
     42  1.1  tsutsui #include <luna68k/dev/psgpam_enc.h>
     43  1.1  tsutsui #include <luna68k/dev/psgpam_table.h>
     44  1.1  tsutsui #else
     45  1.1  tsutsui #include <stdint.h>
     46  1.1  tsutsui #include <stdlib.h>
     47  1.1  tsutsui #include "audio/userland.h"
     48  1.1  tsutsui #include "psgpam_enc.h"
     49  1.1  tsutsui #include "psgpam_table.c"
     50  1.1  tsutsui #include "psgpam_table.h"
     51  1.1  tsutsui #endif
     52  1.1  tsutsui 
     53  1.1  tsutsui void
     54  1.1  tsutsui psgpam_init_context(struct psgpam_codecvar *ctx, u_int sample_rate)
     55  1.1  tsutsui {
     56  1.1  tsutsui 
     57  1.1  tsutsui 	ctx->offset = 65535;
     58  1.1  tsutsui 	ctx->sample_rate = sample_rate;
     59  1.1  tsutsui 	ctx->expire_initial = sample_rate / 10;
     60  1.1  tsutsui 	ctx->expire = ctx->expire_initial;
     61  1.1  tsutsui }
     62  1.1  tsutsui 
     63  1.1  tsutsui static inline auint_t
     64  1.1  tsutsui dynamic_offset(struct psgpam_codecvar *ctx, auint_t v)
     65  1.1  tsutsui {
     66  1.1  tsutsui 
     67  1.1  tsutsui 	/*
     68  1.1  tsutsui 	 * if (the passed value cannot be handled by current offset) {
     69  1.1  tsutsui 	 *   update offset to handle the passed value
     70  1.1  tsutsui 	 * } else {
     71  1.1  tsutsui 	 *   increment offset
     72  1.1  tsutsui 	 * }
     73  1.2  tsutsui 	 */
     74  1.1  tsutsui 	if (v <= ctx->offset) {
     75  1.1  tsutsui 		ctx->offset = v;
     76  1.1  tsutsui 	} else {
     77  1.1  tsutsui 		if (--ctx->expire < 0) {
     78  1.2  tsutsui 			ctx->offset += 1;
     79  1.1  tsutsui 			ctx->expire = ctx->expire_initial;
     80  1.1  tsutsui 		}
     81  1.1  tsutsui 	}
     82  1.1  tsutsui 	return v - ctx->offset;
     83  1.1  tsutsui }
     84  1.1  tsutsui 
     85  1.1  tsutsui #define BULK(table) *d++ = table[v]
     86  1.1  tsutsui 
     87  1.1  tsutsui #define W8(table) *d++ = table[v]
     88  1.1  tsutsui 
     89  1.1  tsutsui #define W16(table) do {							\
     90  1.1  tsutsui 	uint16_t t = (uint16_t)table[v];				\
     91  1.1  tsutsui 	*d++ = ((t & 0xf0) << 4) | (t & 0x0f);				\
     92  1.1  tsutsui } while (0)
     93  1.1  tsutsui 
     94  1.1  tsutsui #define W32(table) do {							\
     95  1.1  tsutsui 	uint32_t t = (uint32_t)table[v];				\
     96  1.1  tsutsui 	*d++ = ((t & 0xf000) << 12)					\
     97  1.1  tsutsui 	     | ((t & 0x0f00) <<  8)					\
     98  1.1  tsutsui 	     | ((t & 0x00f0) <<  4)					\
     99  1.1  tsutsui 	     | ((t & 0x000f));						\
    100  1.1  tsutsui } while (0)
    101  1.1  tsutsui 
    102  1.1  tsutsui #define SPLIT3(table) do {						\
    103  1.1  tsutsui 	uint16_t t = (uint16_t)table[v];				\
    104  1.1  tsutsui 	*d++ = ((t & 0xf000) >> 12);					\
    105  1.1  tsutsui 	*d++ = ((t & 0x0f00) >>  8);					\
    106  1.1  tsutsui 	*d++ = ((t & 0x000f));						\
    107  1.1  tsutsui } while (0)
    108  1.1  tsutsui 
    109  1.3  tsutsui #define ENCODER_DEFINE(enc, TT, table, writer)				\
    110  1.1  tsutsui void									\
    111  1.1  tsutsui psgpam_aint_to_##enc(audio_filter_arg_t *arg)				\
    112  1.1  tsutsui {									\
    113  1.1  tsutsui 	const aint_t *s = arg->src;					\
    114  1.1  tsutsui 	TT *d = arg->dst;						\
    115  1.1  tsutsui 									\
    116  1.1  tsutsui 	for (int i = 0; i < arg->count; i++) {				\
    117  1.1  tsutsui 		auint_t v = (*s++) ^ AINT_T_MIN;			\
    118  1.1  tsutsui 		v >>= (AUDIO_INTERNAL_BITS - table##_BITS);		\
    119  1.1  tsutsui 		writer(table);						\
    120  1.1  tsutsui 	}								\
    121  1.1  tsutsui }
    122  1.1  tsutsui 
    123  1.1  tsutsui #define ENCODER_D_DEFINE(enc, TT, table, writer)			\
    124  1.1  tsutsui void									\
    125  1.1  tsutsui psgpam_aint_to_##enc##_d(audio_filter_arg_t *arg)			\
    126  1.1  tsutsui {									\
    127  1.1  tsutsui 	const aint_t *s = arg->src;					\
    128  1.1  tsutsui 	TT *d = arg->dst;						\
    129  1.1  tsutsui 									\
    130  1.1  tsutsui 	for (int i = 0; i < arg->count; i++) {				\
    131  1.1  tsutsui 		auint_t v = (*s++) ^ AINT_T_MIN;			\
    132  1.1  tsutsui 		v >>= (AUDIO_INTERNAL_BITS - table##_BITS);		\
    133  1.1  tsutsui 		v = dynamic_offset(arg->context, v);			\
    134  1.1  tsutsui 		writer(table);						\
    135  1.1  tsutsui 	}								\
    136  1.1  tsutsui }
    137  1.1  tsutsui 
    138  1.1  tsutsui ENCODER_DEFINE(pam2a, uint16_t, PAM2A_TABLE, W16)
    139  1.1  tsutsui ENCODER_DEFINE(pam2b, uint16_t, PAM2B_TABLE, W16)
    140  1.1  tsutsui ENCODER_DEFINE(pam3a, uint32_t, PAM3A_TABLE, W32)
    141  1.1  tsutsui ENCODER_DEFINE(pam3b, uint32_t, PAM3B_TABLE, W32)
    142  1.1  tsutsui ENCODER_DEFINE(pcm1, uint8_t,  PCM1_TABLE, W8)
    143  1.1  tsutsui ENCODER_DEFINE(pcm2, uint16_t, PCM2_TABLE, W16)
    144  1.1  tsutsui ENCODER_DEFINE(pcm3, uint8_t,  PCM3_TABLE, SPLIT3)
    145  1.1  tsutsui 
    146  1.1  tsutsui ENCODER_D_DEFINE(pam2a, uint16_t, PAM2A_TABLE, W16)
    147  1.1  tsutsui ENCODER_D_DEFINE(pam2b, uint16_t, PAM2B_TABLE, W16)
    148  1.1  tsutsui ENCODER_D_DEFINE(pam3a, uint32_t, PAM3A_TABLE, W32)
    149  1.1  tsutsui ENCODER_D_DEFINE(pam3b, uint32_t, PAM3B_TABLE, W32)
    150  1.1  tsutsui ENCODER_D_DEFINE(pcm1, uint8_t,  PCM1_TABLE, W8)
    151  1.1  tsutsui ENCODER_D_DEFINE(pcm2, uint16_t, PCM2_TABLE, W16)
    152  1.1  tsutsui ENCODER_D_DEFINE(pcm3, uint8_t,  PCM3_TABLE, SPLIT3)
    153