blob: 14f913902fb5862e891b9c9e43e0f5899dba91e8 [file] [log] [blame]
Josh Coalson26560dd2001-02-08 00:38:41 +00001/* libFLAC - Free Lossless Audio Codec library
Josh Coalson305ae2e2002-01-26 17:36:39 +00002 * Copyright (C) 2000,2001,2002 Josh Coalson
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00003 *
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
8 *
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
13 *
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
18 */
19
Josh Coalsonbb7f6b92000-12-10 04:09:52 +000020#include <stdio.h>
21#include <stdlib.h> /* for malloc() */
22#include <string.h> /* for memcpy() */
Josh Coalson1b689822001-05-31 20:11:02 +000023#include "FLAC/assert.h"
Josh Coalsond86e03b2002-08-03 21:56:15 +000024#include "FLAC/stream_decoder.h"
Josh Coalson0a15c142001-06-13 17:59:57 +000025#include "protected/stream_encoder.h"
Josh Coalsonbb7f6b92000-12-10 04:09:52 +000026#include "private/bitbuffer.h"
Josh Coalsoneef56702001-03-30 00:45:22 +000027#include "private/bitmath.h"
Josh Coalson215af572001-03-27 01:15:58 +000028#include "private/crc.h"
Josh Coalsoncf30f502001-05-23 20:57:44 +000029#include "private/cpu.h"
Josh Coalson0a15c142001-06-13 17:59:57 +000030#include "private/stream_encoder_framing.h"
Josh Coalsonbb7f6b92000-12-10 04:09:52 +000031#include "private/fixed.h"
32#include "private/lpc.h"
Josh Coalsonfa37f1c2001-01-12 23:55:11 +000033#include "private/md5.h"
Josh Coalsond98c43d2001-05-13 05:17:01 +000034#include "private/memory.h"
Josh Coalsonbb7f6b92000-12-10 04:09:52 +000035
36#ifdef min
37#undef min
38#endif
39#define min(x,y) ((x)<(y)?(x):(y))
40
41#ifdef max
42#undef max
43#endif
44#define max(x,y) ((x)>(y)?(x):(y))
45
Josh Coalsond86e03b2002-08-03 21:56:15 +000046typedef struct {
47 FLAC__int32 *data[FLAC__MAX_CHANNELS];
48 unsigned size; /* of each data[] in samples */
49 unsigned tail;
50} verify_input_fifo;
51
52typedef struct {
53 const FLAC__byte *data;
54 unsigned capacity;
55 unsigned bytes;
56} verify_output;
57
58typedef enum {
59 ENCODER_IN_MAGIC = 0,
60 ENCODER_IN_METADATA = 1,
61 ENCODER_IN_AUDIO = 2
62} EncoderStateHint;
63
64/*@@@@ function for getting the error_stats */
65
Josh Coalson0a15c142001-06-13 17:59:57 +000066/***********************************************************************
67 *
68 * Private class method prototypes
69 *
70 ***********************************************************************/
71
Josh Coalsonf1eff452002-07-31 07:05:33 +000072static void set_defaults_(FLAC__StreamEncoder *encoder);
73static void free_(FLAC__StreamEncoder *encoder);
74static FLAC__bool resize_buffers_(FLAC__StreamEncoder *encoder, unsigned new_size);
Josh Coalsond86e03b2002-08-03 21:56:15 +000075static FLAC__bool write_bitbuffer_(FLAC__StreamEncoder *encoder, unsigned samples);
Josh Coalsonf1eff452002-07-31 07:05:33 +000076static FLAC__bool process_frame_(FLAC__StreamEncoder *encoder, FLAC__bool is_last_frame);
77static FLAC__bool process_subframes_(FLAC__StreamEncoder *encoder, FLAC__bool is_last_frame);
78static FLAC__bool process_subframe_(FLAC__StreamEncoder *encoder, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool verbatim_only, const FLAC__FrameHeader *frame_header, unsigned subframe_bps, const FLAC__int32 integer_signal[], const FLAC__real real_signal[], FLAC__Subframe *subframe[2], FLAC__int32 *residual[2], unsigned *best_subframe, unsigned *best_bits);
79static FLAC__bool add_subframe_(FLAC__StreamEncoder *encoder, const FLAC__FrameHeader *frame_header, unsigned subframe_bps, const FLAC__Subframe *subframe, FLAC__BitBuffer *frame);
80static unsigned evaluate_constant_subframe_(const FLAC__int32 signal, unsigned subframe_bps, FLAC__Subframe *subframe);
81static unsigned evaluate_fixed_subframe_(FLAC__StreamEncoder *encoder, const FLAC__int32 signal[], FLAC__int32 residual[], FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned raw_bits_per_partition[], unsigned blocksize, unsigned subframe_bps, unsigned order, unsigned rice_parameter, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool do_escape_coding, unsigned rice_parameter_search_dist, FLAC__Subframe *subframe);
82static unsigned evaluate_lpc_subframe_(FLAC__StreamEncoder *encoder, const FLAC__int32 signal[], FLAC__int32 residual[], FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned raw_bits_per_partition[], const FLAC__real lp_coeff[], unsigned blocksize, unsigned subframe_bps, unsigned order, unsigned qlp_coeff_precision, unsigned rice_parameter, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool do_escape_coding, unsigned rice_parameter_search_dist, FLAC__Subframe *subframe);
83static unsigned evaluate_verbatim_subframe_(const FLAC__int32 signal[], unsigned blocksize, unsigned subframe_bps, FLAC__Subframe *subframe);
84static unsigned find_best_partition_order_(struct FLAC__StreamEncoderPrivate *private_, const FLAC__int32 residual[], FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned raw_bits_per_partition[], unsigned residual_samples, unsigned predictor_order, unsigned rice_parameter, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool do_escape_coding, unsigned rice_parameter_search_dist, unsigned *best_partition_order, unsigned best_parameters[], unsigned best_raw_bits[]);
85static void precompute_partition_info_sums_(const FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned residual_samples, unsigned predictor_order, unsigned min_partition_order, unsigned max_partition_order);
86static void precompute_partition_info_escapes_(const FLAC__int32 residual[], unsigned raw_bits_per_partition[], unsigned residual_samples, unsigned predictor_order, unsigned min_partition_order, unsigned max_partition_order);
Josh Coalson8395d022001-07-12 21:25:22 +000087#ifdef DONT_ESTIMATE_RICE_BITS
Josh Coalsonf1eff452002-07-31 07:05:33 +000088static FLAC__bool set_partitioned_rice_(const FLAC__uint32 abs_residual[], const FLAC__int32 residual[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, unsigned parameters[], unsigned *bits);
89static FLAC__bool set_partitioned_rice_with_precompute_(const FLAC__int32 residual[], const FLAC__uint64 abs_residual_partition_sums[], const unsigned raw_bits_per_partition[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, const FLAC__bool search_for_escapes, unsigned parameters[], unsigned raw_bits[], unsigned *bits);
Josh Coalson8395d022001-07-12 21:25:22 +000090#else
Josh Coalsonf1eff452002-07-31 07:05:33 +000091static FLAC__bool set_partitioned_rice_(const FLAC__uint32 abs_residual[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, unsigned parameters[], unsigned *bits);
92static FLAC__bool set_partitioned_rice_with_precompute_(const FLAC__uint32 abs_residual[], const FLAC__uint64 abs_residual_partition_sums[], const unsigned raw_bits_per_partition[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, const FLAC__bool search_for_escapes, unsigned parameters[], unsigned raw_bits[], unsigned *bits);
Josh Coalson0a15c142001-06-13 17:59:57 +000093#endif
Josh Coalsonf1eff452002-07-31 07:05:33 +000094static unsigned get_wasted_bits_(FLAC__int32 signal[], unsigned samples);
Josh Coalsond86e03b2002-08-03 21:56:15 +000095/* verify-related routines: */
96static void append_to_verify_fifo_(verify_input_fifo *fifo, const FLAC__int32 * const input[], unsigned input_offset, unsigned channels, unsigned wide_samples);
97static void append_to_verify_fifo_interleaved_(verify_input_fifo *fifo, const FLAC__int32 input[], unsigned input_offset, unsigned channels, unsigned wide_samples);
98static FLAC__StreamDecoderReadStatus verify_read_callback_(const FLAC__StreamDecoder *decoder, FLAC__byte buffer[], unsigned *bytes, void *client_data);
99static FLAC__StreamDecoderWriteStatus verify_write_callback_(const FLAC__StreamDecoder *decoder, const FLAC__Frame *frame, const FLAC__int32 * const buffer[], void *client_data);
100static void verify_metadata_callback_(const FLAC__StreamDecoder *decoder, const FLAC__StreamMetadata *metadata, void *client_data);
101static void verify_error_callback_(const FLAC__StreamDecoder *decoder, FLAC__StreamDecoderErrorStatus status, void *client_data);
Josh Coalson0a15c142001-06-13 17:59:57 +0000102
103/***********************************************************************
104 *
105 * Private class data
106 *
107 ***********************************************************************/
108
109typedef struct FLAC__StreamEncoderPrivate {
Josh Coalson8395d022001-07-12 21:25:22 +0000110 unsigned input_capacity; /* current size (in samples) of the signal and residual buffers */
Josh Coalson77e3f312001-06-23 03:03:24 +0000111 FLAC__int32 *integer_signal[FLAC__MAX_CHANNELS]; /* the integer version of the input signal */
112 FLAC__int32 *integer_signal_mid_side[2]; /* the integer version of the mid-side input signal (stereo only) */
113 FLAC__real *real_signal[FLAC__MAX_CHANNELS]; /* the floating-point version of the input signal */
114 FLAC__real *real_signal_mid_side[2]; /* the floating-point version of the mid-side input signal (stereo only) */
Josh Coalson8395d022001-07-12 21:25:22 +0000115 unsigned subframe_bps[FLAC__MAX_CHANNELS]; /* the effective bits per sample of the input signal (stream bps - wasted bits) */
116 unsigned subframe_bps_mid_side[2]; /* the effective bits per sample of the mid-side input signal (stream bps - wasted bits + 0/1) */
Josh Coalson77e3f312001-06-23 03:03:24 +0000117 FLAC__int32 *residual_workspace[FLAC__MAX_CHANNELS][2]; /* each channel has a candidate and best workspace where the subframe residual signals will be stored */
118 FLAC__int32 *residual_workspace_mid_side[2][2];
Josh Coalson94e02cd2001-01-25 10:41:06 +0000119 FLAC__Subframe subframe_workspace[FLAC__MAX_CHANNELS][2];
120 FLAC__Subframe subframe_workspace_mid_side[2][2];
121 FLAC__Subframe *subframe_workspace_ptr[FLAC__MAX_CHANNELS][2];
122 FLAC__Subframe *subframe_workspace_ptr_mid_side[2][2];
Josh Coalson8395d022001-07-12 21:25:22 +0000123 unsigned best_subframe[FLAC__MAX_CHANNELS]; /* index into the above workspaces */
Josh Coalson94e02cd2001-01-25 10:41:06 +0000124 unsigned best_subframe_mid_side[2];
Josh Coalson8395d022001-07-12 21:25:22 +0000125 unsigned best_subframe_bits[FLAC__MAX_CHANNELS]; /* size in bits of the best subframe for each channel */
Josh Coalson94e02cd2001-01-25 10:41:06 +0000126 unsigned best_subframe_bits_mid_side[2];
Josh Coalson77e3f312001-06-23 03:03:24 +0000127 FLAC__uint32 *abs_residual; /* workspace where abs(candidate residual) is stored */
Josh Coalsonb3347bd2001-07-16 18:06:41 +0000128 FLAC__uint64 *abs_residual_partition_sums; /* workspace where the sum of abs(candidate residual) for each partition is stored */
Josh Coalson8395d022001-07-12 21:25:22 +0000129 unsigned *raw_bits_per_partition; /* workspace where the sum of silog2(candidate residual) for each partition is stored */
Josh Coalsonaec256b2002-03-12 16:19:54 +0000130 FLAC__BitBuffer *frame; /* the current frame being worked on */
Josh Coalson8395d022001-07-12 21:25:22 +0000131 double loose_mid_side_stereo_frames_exact; /* exact number of frames the encoder will use before trying both independent and mid/side frames again */
132 unsigned loose_mid_side_stereo_frames; /* rounded number of frames the encoder will use before trying both independent and mid/side frames again */
133 unsigned loose_mid_side_stereo_frame_count; /* number of frames using the current channel assignment */
Josh Coalsonb5e60e52001-01-28 09:27:27 +0000134 FLAC__ChannelAssignment last_channel_assignment;
Josh Coalsoncc682512002-06-08 04:53:42 +0000135 FLAC__StreamMetadata metadata;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000136 unsigned current_sample_number;
137 unsigned current_frame_number;
Josh Coalsonfa37f1c2001-01-12 23:55:11 +0000138 struct MD5Context md5context;
Josh Coalsoncf30f502001-05-23 20:57:44 +0000139 FLAC__CPUInfo cpuinfo;
Josh Coalson77e3f312001-06-23 03:03:24 +0000140 unsigned (*local_fixed_compute_best_predictor)(const FLAC__int32 data[], unsigned data_len, FLAC__real residual_bits_per_sample[FLAC__MAX_FIXED_ORDER+1]);
141 void (*local_lpc_compute_autocorrelation)(const FLAC__real data[], unsigned data_len, unsigned lag, FLAC__real autoc[]);
142 void (*local_lpc_compute_residual_from_qlp_coefficients)(const FLAC__int32 data[], unsigned data_len, const FLAC__int32 qlp_coeff[], unsigned order, int lp_quantization, FLAC__int32 residual[]);
143 void (*local_lpc_compute_residual_from_qlp_coefficients_16bit)(const FLAC__int32 data[], unsigned data_len, const FLAC__int32 qlp_coeff[], unsigned order, int lp_quantization, FLAC__int32 residual[]);
Josh Coalson8395d022001-07-12 21:25:22 +0000144 FLAC__bool use_wide_by_block; /* use slow 64-bit versions of some functions because of the block size */
145 FLAC__bool use_wide_by_partition; /* use slow 64-bit versions of some functions because of the min partition order and blocksize */
146 FLAC__bool use_wide_by_order; /* use slow 64-bit versions of some functions because of the lpc order */
147 FLAC__bool precompute_partition_sums; /* our initial guess as to whether precomputing the partitions sums will be a speed improvement */
Josh Coalson681c2932002-08-01 08:19:37 +0000148 FLAC__StreamEncoderWriteCallback write_callback;
149 FLAC__StreamEncoderMetadataCallback metadata_callback;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000150 void *client_data;
Josh Coalsond98c43d2001-05-13 05:17:01 +0000151 /* unaligned (original) pointers to allocated data */
Josh Coalson77e3f312001-06-23 03:03:24 +0000152 FLAC__int32 *integer_signal_unaligned[FLAC__MAX_CHANNELS];
153 FLAC__int32 *integer_signal_mid_side_unaligned[2];
154 FLAC__real *real_signal_unaligned[FLAC__MAX_CHANNELS];
155 FLAC__real *real_signal_mid_side_unaligned[2];
156 FLAC__int32 *residual_workspace_unaligned[FLAC__MAX_CHANNELS][2];
157 FLAC__int32 *residual_workspace_mid_side_unaligned[2][2];
158 FLAC__uint32 *abs_residual_unaligned;
Josh Coalsonb3347bd2001-07-16 18:06:41 +0000159 FLAC__uint64 *abs_residual_partition_sums_unaligned;
Josh Coalsond98c43d2001-05-13 05:17:01 +0000160 unsigned *raw_bits_per_partition_unaligned;
Josh Coalson8084b052001-11-01 00:27:29 +0000161 /*
162 * These fields have been moved here from private function local
163 * declarations merely to save stack space during encoding.
164 */
Josh Coalsonf1eff452002-07-31 07:05:33 +0000165 FLAC__real lp_coeff[FLAC__MAX_LPC_ORDER][FLAC__MAX_LPC_ORDER]; /* from process_subframe_() */
166 unsigned parameters[2][1 << FLAC__MAX_RICE_PARTITION_ORDER], raw_bits[2][1 << FLAC__MAX_RICE_PARTITION_ORDER]; /* from find_best_partition_order_() */
Josh Coalsond86e03b2002-08-03 21:56:15 +0000167 /*
168 * The data for the verify section
169 */
170 struct {
171 FLAC__StreamDecoder *decoder;
172 EncoderStateHint state_hint;
173 FLAC__bool needs_magic_hack;
174 verify_input_fifo input_fifo;
175 verify_output output;
176 struct {
177 FLAC__uint64 absolute_sample;
178 unsigned frame_number;
179 unsigned channel;
180 unsigned sample;
181 FLAC__int32 expected;
182 FLAC__int32 got;
183 } error_stats;
184 } verify;
Josh Coalson0a15c142001-06-13 17:59:57 +0000185} FLAC__StreamEncoderPrivate;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000186
Josh Coalson0a15c142001-06-13 17:59:57 +0000187/***********************************************************************
188 *
189 * Public static class data
190 *
191 ***********************************************************************/
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000192
Josh Coalson57ba6f42002-06-07 05:27:37 +0000193const char * const FLAC__StreamEncoderStateString[] = {
Josh Coalson0a15c142001-06-13 17:59:57 +0000194 "FLAC__STREAM_ENCODER_OK",
Josh Coalsond86e03b2002-08-03 21:56:15 +0000195 "FLAC__STREAM_ENCODER_VERIFY_DECODER_ERROR",
196 "FLAC__STREAM_ENCODER_VERIFY_MISMATCH_IN_AUDIO_DATA",
Josh Coalson00e53872001-06-16 07:32:25 +0000197 "FLAC__STREAM_ENCODER_INVALID_CALLBACK",
Josh Coalson0a15c142001-06-13 17:59:57 +0000198 "FLAC__STREAM_ENCODER_INVALID_NUMBER_OF_CHANNELS",
199 "FLAC__STREAM_ENCODER_INVALID_BITS_PER_SAMPLE",
200 "FLAC__STREAM_ENCODER_INVALID_SAMPLE_RATE",
201 "FLAC__STREAM_ENCODER_INVALID_BLOCK_SIZE",
202 "FLAC__STREAM_ENCODER_INVALID_QLP_COEFF_PRECISION",
203 "FLAC__STREAM_ENCODER_MID_SIDE_CHANNELS_MISMATCH",
204 "FLAC__STREAM_ENCODER_MID_SIDE_SAMPLE_SIZE_MISMATCH",
205 "FLAC__STREAM_ENCODER_ILLEGAL_MID_SIDE_FORCE",
206 "FLAC__STREAM_ENCODER_BLOCK_SIZE_TOO_SMALL_FOR_LPC_ORDER",
207 "FLAC__STREAM_ENCODER_NOT_STREAMABLE",
208 "FLAC__STREAM_ENCODER_FRAMING_ERROR",
Josh Coalson66075c12002-06-01 05:39:38 +0000209 "FLAC__STREAM_ENCODER_INVALID_METADATA",
Josh Coalson0a15c142001-06-13 17:59:57 +0000210 "FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_ENCODING",
211 "FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_WRITING",
212 "FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR",
213 "FLAC__STREAM_ENCODER_ALREADY_INITIALIZED",
214 "FLAC__STREAM_ENCODER_UNINITIALIZED"
Josh Coalsoncbf595f2000-12-22 22:35:33 +0000215};
216
Josh Coalson57ba6f42002-06-07 05:27:37 +0000217const char * const FLAC__StreamEncoderWriteStatusString[] = {
Josh Coalson5c491a12002-08-01 06:39:40 +0000218 "FLAC__STREAM_ENCODER_WRITE_STATUS_OK",
219 "FLAC__STREAM_ENCODER_WRITE_STATUS_FATAL_ERROR"
Josh Coalsoncbf595f2000-12-22 22:35:33 +0000220};
221
Josh Coalson0a15c142001-06-13 17:59:57 +0000222/***********************************************************************
223 *
224 * Class constructor/destructor
225 *
Josh Coalsond86e03b2002-08-03 21:56:15 +0000226 */
Josh Coalson0a15c142001-06-13 17:59:57 +0000227FLAC__StreamEncoder *FLAC__stream_encoder_new()
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000228{
Josh Coalson0a15c142001-06-13 17:59:57 +0000229 FLAC__StreamEncoder *encoder;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000230
Josh Coalson0a15c142001-06-13 17:59:57 +0000231 FLAC__ASSERT(sizeof(int) >= 4); /* we want to die right away if this is not true */
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000232
Josh Coalson0a15c142001-06-13 17:59:57 +0000233 encoder = (FLAC__StreamEncoder*)malloc(sizeof(FLAC__StreamEncoder));
234 if(encoder == 0) {
235 return 0;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000236 }
Josh Coalsond86e03b2002-08-03 21:56:15 +0000237 memset(encoder, 0, sizeof(FLAC__StreamEncoder));
238
Josh Coalsonfa697a92001-08-16 20:07:29 +0000239 encoder->protected_ = (FLAC__StreamEncoderProtected*)malloc(sizeof(FLAC__StreamEncoderProtected));
240 if(encoder->protected_ == 0) {
Josh Coalson0a15c142001-06-13 17:59:57 +0000241 free(encoder);
242 return 0;
Josh Coalsond98c43d2001-05-13 05:17:01 +0000243 }
Josh Coalsond86e03b2002-08-03 21:56:15 +0000244 memset(encoder->protected_, 0, sizeof(FLAC__StreamEncoderProtected));
245
Josh Coalsonfa697a92001-08-16 20:07:29 +0000246 encoder->private_ = (FLAC__StreamEncoderPrivate*)malloc(sizeof(FLAC__StreamEncoderPrivate));
247 if(encoder->private_ == 0) {
248 free(encoder->protected_);
Josh Coalson0a15c142001-06-13 17:59:57 +0000249 free(encoder);
250 return 0;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000251 }
Josh Coalsond86e03b2002-08-03 21:56:15 +0000252 memset(encoder->private_, 0, sizeof(FLAC__StreamEncoderPrivate));
253
Josh Coalsonaec256b2002-03-12 16:19:54 +0000254 encoder->private_->frame = FLAC__bitbuffer_new();
255 if(encoder->private_->frame == 0) {
256 free(encoder->private_);
257 free(encoder->protected_);
258 free(encoder);
259 return 0;
260 }
Josh Coalsond98c43d2001-05-13 05:17:01 +0000261
Josh Coalsonf1eff452002-07-31 07:05:33 +0000262 set_defaults_(encoder);
Josh Coalson92031602002-07-24 06:02:11 +0000263
Josh Coalsonfa697a92001-08-16 20:07:29 +0000264 encoder->protected_->state = FLAC__STREAM_ENCODER_UNINITIALIZED;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000265
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000266 return encoder;
267}
268
Josh Coalson0a15c142001-06-13 17:59:57 +0000269void FLAC__stream_encoder_delete(FLAC__StreamEncoder *encoder)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000270{
Josh Coalsonf1eff452002-07-31 07:05:33 +0000271 FLAC__ASSERT(0 != encoder);
272 FLAC__ASSERT(0 != encoder->protected_);
273 FLAC__ASSERT(0 != encoder->private_);
274 FLAC__ASSERT(0 != encoder->private_->frame);
Josh Coalson0a15c142001-06-13 17:59:57 +0000275
Josh Coalsonf1eff452002-07-31 07:05:33 +0000276 free_(encoder);
Josh Coalsond86e03b2002-08-03 21:56:15 +0000277 if(encoder->protected_->verify && 0 != encoder->private_->verify.decoder)
278 FLAC__stream_decoder_delete(encoder->private_->verify.decoder);
Josh Coalsonaec256b2002-03-12 16:19:54 +0000279 FLAC__bitbuffer_delete(encoder->private_->frame);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000280 free(encoder->private_);
281 free(encoder->protected_);
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000282 free(encoder);
283}
284
Josh Coalson0a15c142001-06-13 17:59:57 +0000285/***********************************************************************
286 *
287 * Public class methods
288 *
289 ***********************************************************************/
290
Josh Coalson00e53872001-06-16 07:32:25 +0000291FLAC__StreamEncoderState FLAC__stream_encoder_init(FLAC__StreamEncoder *encoder)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000292{
293 unsigned i;
294
Josh Coalsonf1eff452002-07-31 07:05:33 +0000295 FLAC__ASSERT(0 != encoder);
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000296
Josh Coalsonfa697a92001-08-16 20:07:29 +0000297 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
298 return encoder->protected_->state = FLAC__STREAM_ENCODER_ALREADY_INITIALIZED;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000299
Josh Coalsonfa697a92001-08-16 20:07:29 +0000300 encoder->protected_->state = FLAC__STREAM_ENCODER_OK;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000301
Josh Coalsonfa697a92001-08-16 20:07:29 +0000302 if(0 == encoder->private_->write_callback || 0 == encoder->private_->metadata_callback)
303 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_CALLBACK;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000304
Josh Coalsonfa697a92001-08-16 20:07:29 +0000305 if(encoder->protected_->channels == 0 || encoder->protected_->channels > FLAC__MAX_CHANNELS)
306 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_NUMBER_OF_CHANNELS;
Josh Coalson69f1ee02001-01-24 00:54:43 +0000307
Josh Coalsonfa697a92001-08-16 20:07:29 +0000308 if(encoder->protected_->do_mid_side_stereo && encoder->protected_->channels != 2)
309 return encoder->protected_->state = FLAC__STREAM_ENCODER_MID_SIDE_CHANNELS_MISMATCH;
Josh Coalsond37d1352001-05-30 23:09:31 +0000310
Josh Coalsonfa697a92001-08-16 20:07:29 +0000311 if(encoder->protected_->loose_mid_side_stereo && !encoder->protected_->do_mid_side_stereo)
312 return encoder->protected_->state = FLAC__STREAM_ENCODER_ILLEGAL_MID_SIDE_FORCE;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000313
Josh Coalsonfa697a92001-08-16 20:07:29 +0000314 if(encoder->protected_->bits_per_sample >= 32)
315 encoder->protected_->do_mid_side_stereo = false; /* since we do 32-bit math, the side channel would have 33 bps and overflow */
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000316
Josh Coalson76c68bc2002-05-17 06:22:02 +0000317 if(encoder->protected_->bits_per_sample < FLAC__MIN_BITS_PER_SAMPLE || encoder->protected_->bits_per_sample > FLAC__REFERENCE_CODEC_MAX_BITS_PER_SAMPLE)
Josh Coalsonfa697a92001-08-16 20:07:29 +0000318 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_BITS_PER_SAMPLE;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000319
Josh Coalson0833f342002-07-15 05:31:55 +0000320 if(!FLAC__format_sample_rate_is_valid(encoder->protected_->sample_rate))
Josh Coalsonfa697a92001-08-16 20:07:29 +0000321 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_SAMPLE_RATE;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000322
Josh Coalsonfa697a92001-08-16 20:07:29 +0000323 if(encoder->protected_->blocksize < FLAC__MIN_BLOCK_SIZE || encoder->protected_->blocksize > FLAC__MAX_BLOCK_SIZE)
324 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_BLOCK_SIZE;
Josh Coalson0a15c142001-06-13 17:59:57 +0000325
Josh Coalsonfa697a92001-08-16 20:07:29 +0000326 if(encoder->protected_->blocksize < encoder->protected_->max_lpc_order)
327 return encoder->protected_->state = FLAC__STREAM_ENCODER_BLOCK_SIZE_TOO_SMALL_FOR_LPC_ORDER;
Josh Coalson0a15c142001-06-13 17:59:57 +0000328
Josh Coalsonfa697a92001-08-16 20:07:29 +0000329 if(encoder->protected_->qlp_coeff_precision == 0) {
330 if(encoder->protected_->bits_per_sample < 16) {
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000331 /* @@@ need some data about how to set this here w.r.t. blocksize and sample rate */
332 /* @@@ until then we'll make a guess */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000333 encoder->protected_->qlp_coeff_precision = max(5, 2 + encoder->protected_->bits_per_sample / 2);
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000334 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000335 else if(encoder->protected_->bits_per_sample == 16) {
336 if(encoder->protected_->blocksize <= 192)
337 encoder->protected_->qlp_coeff_precision = 7;
338 else if(encoder->protected_->blocksize <= 384)
339 encoder->protected_->qlp_coeff_precision = 8;
340 else if(encoder->protected_->blocksize <= 576)
341 encoder->protected_->qlp_coeff_precision = 9;
342 else if(encoder->protected_->blocksize <= 1152)
343 encoder->protected_->qlp_coeff_precision = 10;
344 else if(encoder->protected_->blocksize <= 2304)
345 encoder->protected_->qlp_coeff_precision = 11;
346 else if(encoder->protected_->blocksize <= 4608)
347 encoder->protected_->qlp_coeff_precision = 12;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000348 else
Josh Coalsonfa697a92001-08-16 20:07:29 +0000349 encoder->protected_->qlp_coeff_precision = 13;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000350 }
351 else {
Josh Coalsonfa697a92001-08-16 20:07:29 +0000352 encoder->protected_->qlp_coeff_precision = min(13, 8*sizeof(FLAC__int32) - encoder->protected_->bits_per_sample - 1 - 2); /* @@@ -2 to keep things 32-bit safe */
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000353 }
354 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000355 else if(encoder->protected_->qlp_coeff_precision < FLAC__MIN_QLP_COEFF_PRECISION || encoder->protected_->qlp_coeff_precision + encoder->protected_->bits_per_sample >= 8*sizeof(FLAC__uint32) || encoder->protected_->qlp_coeff_precision >= (1u<<FLAC__SUBFRAME_LPC_QLP_COEFF_PRECISION_LEN))
356 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_QLP_COEFF_PRECISION;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000357
Josh Coalsonfa697a92001-08-16 20:07:29 +0000358 if(encoder->protected_->streamable_subset) {
Josh Coalsonb3347bd2001-07-16 18:06:41 +0000359 /*@@@ add check for blocksize here */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000360 if(encoder->protected_->bits_per_sample != 8 && encoder->protected_->bits_per_sample != 12 && encoder->protected_->bits_per_sample != 16 && encoder->protected_->bits_per_sample != 20 && encoder->protected_->bits_per_sample != 24)
361 return encoder->protected_->state = FLAC__STREAM_ENCODER_NOT_STREAMABLE;
362 if(encoder->protected_->sample_rate > 655350)
363 return encoder->protected_->state = FLAC__STREAM_ENCODER_NOT_STREAMABLE;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000364 }
365
Josh Coalsonfa697a92001-08-16 20:07:29 +0000366 if(encoder->protected_->max_residual_partition_order >= (1u << FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ORDER_LEN))
367 encoder->protected_->max_residual_partition_order = (1u << FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ORDER_LEN) - 1;
368 if(encoder->protected_->min_residual_partition_order >= encoder->protected_->max_residual_partition_order)
369 encoder->protected_->min_residual_partition_order = encoder->protected_->max_residual_partition_order;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000370
Josh Coalson66075c12002-06-01 05:39:38 +0000371 /* validate metadata */
372 if(0 == encoder->protected_->metadata && encoder->protected_->num_metadata_blocks > 0)
373 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_METADATA;
374 for(i = 0; i < encoder->protected_->num_metadata_blocks; i++) {
375 if(encoder->protected_->metadata[i]->type == FLAC__METADATA_TYPE_STREAMINFO)
376 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_METADATA;
377 else if(encoder->protected_->metadata[i]->type == FLAC__METADATA_TYPE_SEEKTABLE) {
Josh Coalson0833f342002-07-15 05:31:55 +0000378 if(!FLAC__format_seektable_is_legal(&encoder->protected_->metadata[i]->data.seek_table))
Josh Coalson66075c12002-06-01 05:39:38 +0000379 return encoder->protected_->state = FLAC__STREAM_ENCODER_INVALID_METADATA;
380 }
381 }
382
Josh Coalsonfa697a92001-08-16 20:07:29 +0000383 encoder->private_->input_capacity = 0;
384 for(i = 0; i < encoder->protected_->channels; i++) {
385 encoder->private_->integer_signal_unaligned[i] = encoder->private_->integer_signal[i] = 0;
386 encoder->private_->real_signal_unaligned[i] = encoder->private_->real_signal[i] = 0;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000387 }
388 for(i = 0; i < 2; i++) {
Josh Coalsonfa697a92001-08-16 20:07:29 +0000389 encoder->private_->integer_signal_mid_side_unaligned[i] = encoder->private_->integer_signal_mid_side[i] = 0;
390 encoder->private_->real_signal_mid_side_unaligned[i] = encoder->private_->real_signal_mid_side[i] = 0;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000391 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000392 for(i = 0; i < encoder->protected_->channels; i++) {
393 encoder->private_->residual_workspace_unaligned[i][0] = encoder->private_->residual_workspace[i][0] = 0;
394 encoder->private_->residual_workspace_unaligned[i][1] = encoder->private_->residual_workspace[i][1] = 0;
395 encoder->private_->best_subframe[i] = 0;
Josh Coalson94e02cd2001-01-25 10:41:06 +0000396 }
397 for(i = 0; i < 2; i++) {
Josh Coalsonfa697a92001-08-16 20:07:29 +0000398 encoder->private_->residual_workspace_mid_side_unaligned[i][0] = encoder->private_->residual_workspace_mid_side[i][0] = 0;
399 encoder->private_->residual_workspace_mid_side_unaligned[i][1] = encoder->private_->residual_workspace_mid_side[i][1] = 0;
400 encoder->private_->best_subframe_mid_side[i] = 0;
Josh Coalson94e02cd2001-01-25 10:41:06 +0000401 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000402 for(i = 0; i < encoder->protected_->channels; i++) {
403 encoder->private_->subframe_workspace_ptr[i][0] = &encoder->private_->subframe_workspace[i][0];
404 encoder->private_->subframe_workspace_ptr[i][1] = &encoder->private_->subframe_workspace[i][1];
Josh Coalson94e02cd2001-01-25 10:41:06 +0000405 }
406 for(i = 0; i < 2; i++) {
Josh Coalsonfa697a92001-08-16 20:07:29 +0000407 encoder->private_->subframe_workspace_ptr_mid_side[i][0] = &encoder->private_->subframe_workspace_mid_side[i][0];
408 encoder->private_->subframe_workspace_ptr_mid_side[i][1] = &encoder->private_->subframe_workspace_mid_side[i][1];
Josh Coalson94e02cd2001-01-25 10:41:06 +0000409 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000410 encoder->private_->abs_residual_unaligned = encoder->private_->abs_residual = 0;
411 encoder->private_->abs_residual_partition_sums_unaligned = encoder->private_->abs_residual_partition_sums = 0;
412 encoder->private_->raw_bits_per_partition_unaligned = encoder->private_->raw_bits_per_partition = 0;
413 encoder->private_->loose_mid_side_stereo_frames_exact = (double)encoder->protected_->sample_rate * 0.4 / (double)encoder->protected_->blocksize;
414 encoder->private_->loose_mid_side_stereo_frames = (unsigned)(encoder->private_->loose_mid_side_stereo_frames_exact + 0.5);
415 if(encoder->private_->loose_mid_side_stereo_frames == 0)
416 encoder->private_->loose_mid_side_stereo_frames = 1;
417 encoder->private_->loose_mid_side_stereo_frame_count = 0;
418 encoder->private_->current_sample_number = 0;
419 encoder->private_->current_frame_number = 0;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000420
Josh Coalsonfa697a92001-08-16 20:07:29 +0000421 encoder->private_->use_wide_by_block = (encoder->protected_->bits_per_sample + FLAC__bitmath_ilog2(encoder->protected_->blocksize)+1 > 30);
422 encoder->private_->use_wide_by_order = (encoder->protected_->bits_per_sample + FLAC__bitmath_ilog2(max(encoder->protected_->max_lpc_order, FLAC__MAX_FIXED_ORDER))+1 > 30); /*@@@ need to use this? */
423 encoder->private_->use_wide_by_partition = (false); /*@@@ need to set this */
Josh Coalson8395d022001-07-12 21:25:22 +0000424
Josh Coalsoncf30f502001-05-23 20:57:44 +0000425 /*
426 * get the CPU info and set the function pointers
427 */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000428 FLAC__cpu_info(&encoder->private_->cpuinfo);
Josh Coalsoncf30f502001-05-23 20:57:44 +0000429 /* first default to the non-asm routines */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000430 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation;
431 encoder->private_->local_fixed_compute_best_predictor = FLAC__fixed_compute_best_predictor;
432 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients = FLAC__lpc_compute_residual_from_qlp_coefficients;
433 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients_16bit = FLAC__lpc_compute_residual_from_qlp_coefficients;
Josh Coalsoncf30f502001-05-23 20:57:44 +0000434 /* now override with asm where appropriate */
Josh Coalsona3f7c2c2001-05-25 00:04:45 +0000435#ifndef FLAC__NO_ASM
Josh Coalsonfa697a92001-08-16 20:07:29 +0000436 if(encoder->private_->cpuinfo.use_asm) {
Josh Coalsoncf30f502001-05-23 20:57:44 +0000437#ifdef FLAC__CPU_IA32
Josh Coalsonfa697a92001-08-16 20:07:29 +0000438 FLAC__ASSERT(encoder->private_->cpuinfo.type == FLAC__CPUINFO_TYPE_IA32);
Josh Coalson034d38e2001-05-24 19:29:30 +0000439#ifdef FLAC__HAS_NASM
Josh Coalsonfa697a92001-08-16 20:07:29 +0000440 if(0 && encoder->private_->cpuinfo.data.ia32.sse) {
441 if(encoder->protected_->max_lpc_order < 4)
442 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation_asm_ia32_sse_lag_4;
443 else if(encoder->protected_->max_lpc_order < 8)
444 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation_asm_ia32_sse_lag_8;
445 else if(encoder->protected_->max_lpc_order < 12)
446 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation_asm_ia32_sse_lag_12;
Josh Coalson021ad3b2001-07-18 00:25:52 +0000447 else
Josh Coalsonfa697a92001-08-16 20:07:29 +0000448 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation_asm_ia32;
Josh Coalson021ad3b2001-07-18 00:25:52 +0000449 }
Josh Coalson395938e2001-11-15 21:53:25 +0000450 else if(encoder->private_->cpuinfo.data.ia32._3dnow)
Josh Coalsonfa697a92001-08-16 20:07:29 +0000451 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation_asm_ia32_3dnow;
Josh Coalsonaa255362001-05-31 06:17:41 +0000452 else
Josh Coalsonfa697a92001-08-16 20:07:29 +0000453 encoder->private_->local_lpc_compute_autocorrelation = FLAC__lpc_compute_autocorrelation_asm_ia32;
454 if(encoder->private_->cpuinfo.data.ia32.mmx && encoder->private_->cpuinfo.data.ia32.cmov)
455 encoder->private_->local_fixed_compute_best_predictor = FLAC__fixed_compute_best_predictor_asm_ia32_mmx_cmov;
456 if(encoder->private_->cpuinfo.data.ia32.mmx) {
457 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients = FLAC__lpc_compute_residual_from_qlp_coefficients_asm_ia32;
458 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients_16bit = FLAC__lpc_compute_residual_from_qlp_coefficients_asm_ia32_mmx;
Josh Coalson021ad3b2001-07-18 00:25:52 +0000459 }
460 else {
Josh Coalsonfa697a92001-08-16 20:07:29 +0000461 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients = FLAC__lpc_compute_residual_from_qlp_coefficients_asm_ia32;
462 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients_16bit = FLAC__lpc_compute_residual_from_qlp_coefficients_asm_ia32;
Josh Coalson021ad3b2001-07-18 00:25:52 +0000463 }
Josh Coalsoncf30f502001-05-23 20:57:44 +0000464#endif
Josh Coalson034d38e2001-05-24 19:29:30 +0000465#endif
Josh Coalson021ad3b2001-07-18 00:25:52 +0000466 }
Josh Coalsona3f7c2c2001-05-25 00:04:45 +0000467#endif
Josh Coalson8395d022001-07-12 21:25:22 +0000468 /* finally override based on wide-ness if necessary */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000469 if(encoder->private_->use_wide_by_block) {
470 encoder->private_->local_fixed_compute_best_predictor = FLAC__fixed_compute_best_predictor_wide;
Josh Coalson8395d022001-07-12 21:25:22 +0000471 }
Josh Coalsoncf30f502001-05-23 20:57:44 +0000472
Josh Coalson8395d022001-07-12 21:25:22 +0000473 /* we require precompute_partition_sums if do_escape_coding because of their intertwined nature */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000474 encoder->private_->precompute_partition_sums = (encoder->protected_->max_residual_partition_order > encoder->protected_->min_residual_partition_order) || encoder->protected_->do_escape_coding;
Josh Coalsoneef56702001-03-30 00:45:22 +0000475
Josh Coalsonf1eff452002-07-31 07:05:33 +0000476 if(!resize_buffers_(encoder, encoder->protected_->blocksize)) {
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000477 /* the above function sets the state for us in case of an error */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000478 return encoder->protected_->state;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000479 }
Josh Coalsonaec256b2002-03-12 16:19:54 +0000480
481 if(!FLAC__bitbuffer_init(encoder->private_->frame))
482 return encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000483
484 /*
Josh Coalsond86e03b2002-08-03 21:56:15 +0000485 * Set up the verify stuff if necessary
486 */
487 if(encoder->protected_->verify) {
488 /*
489 * First, set up the fifo which will hold the
490 * original signal to compare against
491 */
492 encoder->private_->verify.input_fifo.size = encoder->protected_->blocksize;
493 for(i = 0; i < encoder->protected_->channels; i++) {
494 if(0 == (encoder->private_->verify.input_fifo.data[i] = (FLAC__int32*)malloc(sizeof(FLAC__int32) * encoder->private_->verify.input_fifo.size)))
495 return encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
496 }
497 encoder->private_->verify.input_fifo.tail = 0;
498
499 /*
500 * Now set up a stream decoder for verification
501 */
502 encoder->private_->verify.decoder = FLAC__stream_decoder_new();
503 if(0 == encoder->private_->verify.decoder)
504 return encoder->protected_->state = FLAC__STREAM_ENCODER_VERIFY_DECODER_ERROR;
505
506 FLAC__stream_decoder_set_read_callback(encoder->private_->verify.decoder, verify_read_callback_);
507 FLAC__stream_decoder_set_write_callback(encoder->private_->verify.decoder, verify_write_callback_);
508 FLAC__stream_decoder_set_metadata_callback(encoder->private_->verify.decoder, verify_metadata_callback_);
509 FLAC__stream_decoder_set_error_callback(encoder->private_->verify.decoder, verify_error_callback_);
510 FLAC__stream_decoder_set_client_data(encoder->private_->verify.decoder, encoder);
511 if(FLAC__stream_decoder_init(encoder->private_->verify.decoder) != FLAC__STREAM_DECODER_SEARCH_FOR_METADATA)
512 return encoder->protected_->state = FLAC__STREAM_ENCODER_VERIFY_DECODER_ERROR;
513 }
514
515 /*
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000516 * write the stream header
517 */
Josh Coalsond86e03b2002-08-03 21:56:15 +0000518 if(encoder->protected_->verify)
519 encoder->private_->verify.state_hint = ENCODER_IN_MAGIC;
Josh Coalsonaec256b2002-03-12 16:19:54 +0000520 if(!FLAC__bitbuffer_write_raw_uint32(encoder->private_->frame, FLAC__STREAM_SYNC, FLAC__STREAM_SYNC_LEN))
Josh Coalsonfa697a92001-08-16 20:07:29 +0000521 return encoder->protected_->state = FLAC__STREAM_ENCODER_FRAMING_ERROR;
Josh Coalsond86e03b2002-08-03 21:56:15 +0000522 if(!write_bitbuffer_(encoder, 0)) {
523 /* the above function sets the state for us in case of an error */
524 return encoder->protected_->state;
525 }
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000526
Josh Coalson5c491a12002-08-01 06:39:40 +0000527 /*
528 * write the STREAMINFO metadata block
529 */
Josh Coalsond86e03b2002-08-03 21:56:15 +0000530 if(encoder->protected_->verify)
531 encoder->private_->verify.state_hint = ENCODER_IN_METADATA;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000532 encoder->private_->metadata.type = FLAC__METADATA_TYPE_STREAMINFO;
Josh Coalson66075c12002-06-01 05:39:38 +0000533 encoder->private_->metadata.is_last = (encoder->protected_->num_metadata_blocks == 0);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000534 encoder->private_->metadata.length = FLAC__STREAM_METADATA_STREAMINFO_LENGTH;
535 encoder->private_->metadata.data.stream_info.min_blocksize = encoder->protected_->blocksize; /* this encoder uses the same blocksize for the whole stream */
536 encoder->private_->metadata.data.stream_info.max_blocksize = encoder->protected_->blocksize;
537 encoder->private_->metadata.data.stream_info.min_framesize = 0; /* we don't know this yet; have to fill it in later */
538 encoder->private_->metadata.data.stream_info.max_framesize = 0; /* we don't know this yet; have to fill it in later */
539 encoder->private_->metadata.data.stream_info.sample_rate = encoder->protected_->sample_rate;
540 encoder->private_->metadata.data.stream_info.channels = encoder->protected_->channels;
541 encoder->private_->metadata.data.stream_info.bits_per_sample = encoder->protected_->bits_per_sample;
542 encoder->private_->metadata.data.stream_info.total_samples = encoder->protected_->total_samples_estimate; /* we will replace this later with the real total */
543 memset(encoder->private_->metadata.data.stream_info.md5sum, 0, 16); /* we don't know this yet; have to fill it in later */
544 MD5Init(&encoder->private_->md5context);
Josh Coalson5c491a12002-08-01 06:39:40 +0000545 if(!FLAC__bitbuffer_clear(encoder->private_->frame)) {
546 encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
547 return false;
548 }
Josh Coalsonaec256b2002-03-12 16:19:54 +0000549 if(!FLAC__add_metadata_block(&encoder->private_->metadata, encoder->private_->frame))
Josh Coalsonfa697a92001-08-16 20:07:29 +0000550 return encoder->protected_->state = FLAC__STREAM_ENCODER_FRAMING_ERROR;
Josh Coalsond86e03b2002-08-03 21:56:15 +0000551 if(!write_bitbuffer_(encoder, 0)) {
552 /* the above function sets the state for us in case of an error */
553 return encoder->protected_->state;
554 }
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000555
Josh Coalson5c491a12002-08-01 06:39:40 +0000556 /*
557 * Now that the STREAMINFO block is written, we can init this to an
558 * absurdly-high value...
559 */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000560 encoder->private_->metadata.data.stream_info.min_framesize = (1u << FLAC__STREAM_METADATA_STREAMINFO_MIN_FRAME_SIZE_LEN) - 1;
Josh Coalsoncbbbb5f2001-01-23 00:41:48 +0000561 /* ... and clear this to 0 */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000562 encoder->private_->metadata.data.stream_info.total_samples = 0;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000563
Josh Coalson5c491a12002-08-01 06:39:40 +0000564 /*
565 * write the user's metadata blocks
566 */
567 for(i = 0; i < encoder->protected_->num_metadata_blocks; i++) {
568 encoder->protected_->metadata[i]->is_last = (i == encoder->protected_->num_metadata_blocks - 1);
569 if(!FLAC__bitbuffer_clear(encoder->private_->frame)) {
570 encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
571 return false;
572 }
573 if(!FLAC__add_metadata_block(encoder->protected_->metadata[i], encoder->private_->frame))
574 return encoder->protected_->state = FLAC__STREAM_ENCODER_FRAMING_ERROR;
Josh Coalsond86e03b2002-08-03 21:56:15 +0000575 if(!write_bitbuffer_(encoder, 0)) {
576 /* the above function sets the state for us in case of an error */
577 return encoder->protected_->state;
578 }
Josh Coalson5c491a12002-08-01 06:39:40 +0000579 }
580
Josh Coalsond86e03b2002-08-03 21:56:15 +0000581 if(encoder->protected_->verify)
582 encoder->private_->verify.state_hint = ENCODER_IN_AUDIO;
583
Josh Coalsonfa697a92001-08-16 20:07:29 +0000584 return encoder->protected_->state;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000585}
586
Josh Coalson0a15c142001-06-13 17:59:57 +0000587void FLAC__stream_encoder_finish(FLAC__StreamEncoder *encoder)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000588{
Josh Coalsonf1eff452002-07-31 07:05:33 +0000589 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000590 if(encoder->protected_->state == FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000591 return;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000592 if(encoder->private_->current_sample_number != 0) {
593 encoder->protected_->blocksize = encoder->private_->current_sample_number;
Josh Coalsonf1eff452002-07-31 07:05:33 +0000594 process_frame_(encoder, true); /* true => is last frame */
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000595 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000596 MD5Final(encoder->private_->metadata.data.stream_info.md5sum, &encoder->private_->md5context);
597 encoder->private_->metadata_callback(encoder, &encoder->private_->metadata, encoder->private_->client_data);
Josh Coalson0a15c142001-06-13 17:59:57 +0000598
Josh Coalsond86e03b2002-08-03 21:56:15 +0000599 if(encoder->protected_->verify && 0 != encoder->private_->verify.decoder)
600 FLAC__stream_decoder_finish(encoder->private_->verify.decoder);
601
Josh Coalsonf1eff452002-07-31 07:05:33 +0000602 free_(encoder);
603 set_defaults_(encoder);
Josh Coalson92031602002-07-24 06:02:11 +0000604
Josh Coalsonfa697a92001-08-16 20:07:29 +0000605 encoder->protected_->state = FLAC__STREAM_ENCODER_UNINITIALIZED;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000606}
607
Josh Coalsond86e03b2002-08-03 21:56:15 +0000608FLAC__bool FLAC__stream_encoder_set_verify(FLAC__StreamEncoder *encoder, FLAC__bool value)
609{
610 FLAC__ASSERT(0 != encoder);
611 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
612 return false;
613 encoder->protected_->verify = value;
614 return true;
615}
616
Josh Coalson16556042002-05-29 05:51:24 +0000617FLAC__bool FLAC__stream_encoder_set_streamable_subset(FLAC__StreamEncoder *encoder, FLAC__bool value)
Josh Coalson00e53872001-06-16 07:32:25 +0000618{
Josh Coalson92031602002-07-24 06:02:11 +0000619 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000620 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000621 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000622 encoder->protected_->streamable_subset = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000623 return true;
624}
625
Josh Coalson16556042002-05-29 05:51:24 +0000626FLAC__bool FLAC__stream_encoder_set_do_mid_side_stereo(FLAC__StreamEncoder *encoder, FLAC__bool value)
Josh Coalson00e53872001-06-16 07:32:25 +0000627{
Josh Coalson92031602002-07-24 06:02:11 +0000628 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000629 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000630 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000631 encoder->protected_->do_mid_side_stereo = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000632 return true;
633}
634
Josh Coalson16556042002-05-29 05:51:24 +0000635FLAC__bool FLAC__stream_encoder_set_loose_mid_side_stereo(FLAC__StreamEncoder *encoder, FLAC__bool value)
Josh Coalson00e53872001-06-16 07:32:25 +0000636{
Josh Coalson92031602002-07-24 06:02:11 +0000637 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000638 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000639 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000640 encoder->protected_->loose_mid_side_stereo = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000641 return true;
642}
643
Josh Coalson16556042002-05-29 05:51:24 +0000644FLAC__bool FLAC__stream_encoder_set_channels(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000645{
Josh Coalson92031602002-07-24 06:02:11 +0000646 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000647 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000648 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000649 encoder->protected_->channels = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000650 return true;
651}
652
Josh Coalson16556042002-05-29 05:51:24 +0000653FLAC__bool FLAC__stream_encoder_set_bits_per_sample(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000654{
Josh Coalson92031602002-07-24 06:02:11 +0000655 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000656 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000657 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000658 encoder->protected_->bits_per_sample = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000659 return true;
660}
661
Josh Coalson16556042002-05-29 05:51:24 +0000662FLAC__bool FLAC__stream_encoder_set_sample_rate(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000663{
Josh Coalson92031602002-07-24 06:02:11 +0000664 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000665 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000666 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000667 encoder->protected_->sample_rate = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000668 return true;
669}
670
Josh Coalson16556042002-05-29 05:51:24 +0000671FLAC__bool FLAC__stream_encoder_set_blocksize(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000672{
Josh Coalson92031602002-07-24 06:02:11 +0000673 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000674 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000675 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000676 encoder->protected_->blocksize = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000677 return true;
678}
679
Josh Coalson16556042002-05-29 05:51:24 +0000680FLAC__bool FLAC__stream_encoder_set_max_lpc_order(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000681{
Josh Coalson92031602002-07-24 06:02:11 +0000682 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000683 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000684 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000685 encoder->protected_->max_lpc_order = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000686 return true;
687}
688
Josh Coalson16556042002-05-29 05:51:24 +0000689FLAC__bool FLAC__stream_encoder_set_qlp_coeff_precision(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000690{
Josh Coalson92031602002-07-24 06:02:11 +0000691 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000692 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000693 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000694 encoder->protected_->qlp_coeff_precision = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000695 return true;
696}
697
Josh Coalson16556042002-05-29 05:51:24 +0000698FLAC__bool FLAC__stream_encoder_set_do_qlp_coeff_prec_search(FLAC__StreamEncoder *encoder, FLAC__bool value)
Josh Coalson00e53872001-06-16 07:32:25 +0000699{
Josh Coalson92031602002-07-24 06:02:11 +0000700 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000701 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000702 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000703 encoder->protected_->do_qlp_coeff_prec_search = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000704 return true;
705}
706
Josh Coalson16556042002-05-29 05:51:24 +0000707FLAC__bool FLAC__stream_encoder_set_do_escape_coding(FLAC__StreamEncoder *encoder, FLAC__bool value)
Josh Coalson8395d022001-07-12 21:25:22 +0000708{
Josh Coalson92031602002-07-24 06:02:11 +0000709 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000710 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson8395d022001-07-12 21:25:22 +0000711 return false;
Josh Coalson680e3aa2002-08-01 07:32:17 +0000712#if 0
713 /*@@@ deprecated: */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000714 encoder->protected_->do_escape_coding = value;
Josh Coalson680e3aa2002-08-01 07:32:17 +0000715#else
716 (void)value;
717#endif
Josh Coalson8395d022001-07-12 21:25:22 +0000718 return true;
719}
720
Josh Coalson16556042002-05-29 05:51:24 +0000721FLAC__bool FLAC__stream_encoder_set_do_exhaustive_model_search(FLAC__StreamEncoder *encoder, FLAC__bool value)
Josh Coalson00e53872001-06-16 07:32:25 +0000722{
Josh Coalson92031602002-07-24 06:02:11 +0000723 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000724 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000725 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000726 encoder->protected_->do_exhaustive_model_search = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000727 return true;
728}
729
Josh Coalson16556042002-05-29 05:51:24 +0000730FLAC__bool FLAC__stream_encoder_set_min_residual_partition_order(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000731{
Josh Coalson92031602002-07-24 06:02:11 +0000732 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000733 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000734 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000735 encoder->protected_->min_residual_partition_order = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000736 return true;
737}
738
Josh Coalson16556042002-05-29 05:51:24 +0000739FLAC__bool FLAC__stream_encoder_set_max_residual_partition_order(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000740{
Josh Coalson92031602002-07-24 06:02:11 +0000741 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000742 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000743 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000744 encoder->protected_->max_residual_partition_order = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000745 return true;
746}
747
Josh Coalson16556042002-05-29 05:51:24 +0000748FLAC__bool FLAC__stream_encoder_set_rice_parameter_search_dist(FLAC__StreamEncoder *encoder, unsigned value)
Josh Coalson00e53872001-06-16 07:32:25 +0000749{
Josh Coalson92031602002-07-24 06:02:11 +0000750 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000751 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000752 return false;
Josh Coalson680e3aa2002-08-01 07:32:17 +0000753#if 0
754 /*@@@ deprecated: */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000755 encoder->protected_->rice_parameter_search_dist = value;
Josh Coalson680e3aa2002-08-01 07:32:17 +0000756#else
757 (void)value;
758#endif
Josh Coalson00e53872001-06-16 07:32:25 +0000759 return true;
760}
761
Josh Coalson16556042002-05-29 05:51:24 +0000762FLAC__bool FLAC__stream_encoder_set_total_samples_estimate(FLAC__StreamEncoder *encoder, FLAC__uint64 value)
Josh Coalson00e53872001-06-16 07:32:25 +0000763{
Josh Coalson92031602002-07-24 06:02:11 +0000764 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000765 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000766 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000767 encoder->protected_->total_samples_estimate = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000768 return true;
769}
770
Josh Coalsoncc682512002-06-08 04:53:42 +0000771FLAC__bool FLAC__stream_encoder_set_metadata(FLAC__StreamEncoder *encoder, FLAC__StreamMetadata **metadata, unsigned num_blocks)
Josh Coalson00e53872001-06-16 07:32:25 +0000772{
Josh Coalson92031602002-07-24 06:02:11 +0000773 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000774 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000775 return false;
Josh Coalson66075c12002-06-01 05:39:38 +0000776 encoder->protected_->metadata = metadata;
777 encoder->protected_->num_metadata_blocks = num_blocks;
Josh Coalson00e53872001-06-16 07:32:25 +0000778 return true;
779}
780
Josh Coalson681c2932002-08-01 08:19:37 +0000781FLAC__bool FLAC__stream_encoder_set_write_callback(FLAC__StreamEncoder *encoder, FLAC__StreamEncoderWriteCallback value)
Josh Coalson00e53872001-06-16 07:32:25 +0000782{
Josh Coalson92031602002-07-24 06:02:11 +0000783 FLAC__ASSERT(0 != encoder);
784 FLAC__ASSERT(0 != value);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000785 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000786 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000787 encoder->private_->write_callback = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000788 return true;
789}
790
Josh Coalson681c2932002-08-01 08:19:37 +0000791FLAC__bool FLAC__stream_encoder_set_metadata_callback(FLAC__StreamEncoder *encoder, FLAC__StreamEncoderMetadataCallback value)
Josh Coalson00e53872001-06-16 07:32:25 +0000792{
Josh Coalson92031602002-07-24 06:02:11 +0000793 FLAC__ASSERT(0 != encoder);
794 FLAC__ASSERT(0 != value);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000795 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000796 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000797 encoder->private_->metadata_callback = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000798 return true;
799}
800
Josh Coalson16556042002-05-29 05:51:24 +0000801FLAC__bool FLAC__stream_encoder_set_client_data(FLAC__StreamEncoder *encoder, void *value)
Josh Coalson00e53872001-06-16 07:32:25 +0000802{
Josh Coalson92031602002-07-24 06:02:11 +0000803 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000804 if(encoder->protected_->state != FLAC__STREAM_ENCODER_UNINITIALIZED)
Josh Coalson00e53872001-06-16 07:32:25 +0000805 return false;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000806 encoder->private_->client_data = value;
Josh Coalson00e53872001-06-16 07:32:25 +0000807 return true;
808}
809
810FLAC__StreamEncoderState FLAC__stream_encoder_get_state(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000811{
Josh Coalson92031602002-07-24 06:02:11 +0000812 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000813 return encoder->protected_->state;
Josh Coalson0a15c142001-06-13 17:59:57 +0000814}
815
Josh Coalsond86e03b2002-08-03 21:56:15 +0000816FLAC__StreamDecoderState FLAC__stream_encoder_get_verify_decoder_state(const FLAC__StreamEncoder *encoder)
817{
818 FLAC__ASSERT(0 != encoder);
819 if(encoder->protected_->verify)
820 return FLAC__stream_decoder_get_state(encoder->private_->verify.decoder);
821 else
822 return FLAC__STREAM_DECODER_UNINITIALIZED;
823}
824
825FLAC__bool FLAC__stream_encoder_get_verify(const FLAC__StreamEncoder *encoder)
826{
827 FLAC__ASSERT(0 != encoder);
828 return encoder->protected_->verify;
829}
830
Josh Coalson77e3f312001-06-23 03:03:24 +0000831FLAC__bool FLAC__stream_encoder_get_streamable_subset(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000832{
Josh Coalson92031602002-07-24 06:02:11 +0000833 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000834 return encoder->protected_->streamable_subset;
Josh Coalson0a15c142001-06-13 17:59:57 +0000835}
836
Josh Coalson77e3f312001-06-23 03:03:24 +0000837FLAC__bool FLAC__stream_encoder_get_do_mid_side_stereo(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000838{
Josh Coalson92031602002-07-24 06:02:11 +0000839 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000840 return encoder->protected_->do_mid_side_stereo;
Josh Coalson0a15c142001-06-13 17:59:57 +0000841}
842
Josh Coalson77e3f312001-06-23 03:03:24 +0000843FLAC__bool FLAC__stream_encoder_get_loose_mid_side_stereo(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000844{
Josh Coalson92031602002-07-24 06:02:11 +0000845 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000846 return encoder->protected_->loose_mid_side_stereo;
Josh Coalson0a15c142001-06-13 17:59:57 +0000847}
848
Josh Coalson00e53872001-06-16 07:32:25 +0000849unsigned FLAC__stream_encoder_get_channels(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000850{
Josh Coalson92031602002-07-24 06:02:11 +0000851 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000852 return encoder->protected_->channels;
Josh Coalson0a15c142001-06-13 17:59:57 +0000853}
854
Josh Coalson00e53872001-06-16 07:32:25 +0000855unsigned FLAC__stream_encoder_get_bits_per_sample(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000856{
Josh Coalson92031602002-07-24 06:02:11 +0000857 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000858 return encoder->protected_->bits_per_sample;
Josh Coalson0a15c142001-06-13 17:59:57 +0000859}
860
Josh Coalson00e53872001-06-16 07:32:25 +0000861unsigned FLAC__stream_encoder_get_sample_rate(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000862{
Josh Coalson92031602002-07-24 06:02:11 +0000863 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000864 return encoder->protected_->sample_rate;
Josh Coalson0a15c142001-06-13 17:59:57 +0000865}
866
Josh Coalson00e53872001-06-16 07:32:25 +0000867unsigned FLAC__stream_encoder_get_blocksize(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000868{
Josh Coalson92031602002-07-24 06:02:11 +0000869 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000870 return encoder->protected_->blocksize;
Josh Coalson0a15c142001-06-13 17:59:57 +0000871}
872
Josh Coalson00e53872001-06-16 07:32:25 +0000873unsigned FLAC__stream_encoder_get_max_lpc_order(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000874{
Josh Coalson92031602002-07-24 06:02:11 +0000875 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000876 return encoder->protected_->max_lpc_order;
Josh Coalson0a15c142001-06-13 17:59:57 +0000877}
878
Josh Coalson00e53872001-06-16 07:32:25 +0000879unsigned FLAC__stream_encoder_get_qlp_coeff_precision(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000880{
Josh Coalson92031602002-07-24 06:02:11 +0000881 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000882 return encoder->protected_->qlp_coeff_precision;
Josh Coalson0a15c142001-06-13 17:59:57 +0000883}
884
Josh Coalson77e3f312001-06-23 03:03:24 +0000885FLAC__bool FLAC__stream_encoder_get_do_qlp_coeff_prec_search(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000886{
Josh Coalson92031602002-07-24 06:02:11 +0000887 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000888 return encoder->protected_->do_qlp_coeff_prec_search;
Josh Coalson0a15c142001-06-13 17:59:57 +0000889}
890
Josh Coalson8395d022001-07-12 21:25:22 +0000891FLAC__bool FLAC__stream_encoder_get_do_escape_coding(const FLAC__StreamEncoder *encoder)
892{
Josh Coalson92031602002-07-24 06:02:11 +0000893 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000894 return encoder->protected_->do_escape_coding;
Josh Coalson8395d022001-07-12 21:25:22 +0000895}
896
Josh Coalson77e3f312001-06-23 03:03:24 +0000897FLAC__bool FLAC__stream_encoder_get_do_exhaustive_model_search(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000898{
Josh Coalson92031602002-07-24 06:02:11 +0000899 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000900 return encoder->protected_->do_exhaustive_model_search;
Josh Coalson0a15c142001-06-13 17:59:57 +0000901}
902
Josh Coalson00e53872001-06-16 07:32:25 +0000903unsigned FLAC__stream_encoder_get_min_residual_partition_order(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000904{
Josh Coalson92031602002-07-24 06:02:11 +0000905 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000906 return encoder->protected_->min_residual_partition_order;
Josh Coalson0a15c142001-06-13 17:59:57 +0000907}
908
Josh Coalson00e53872001-06-16 07:32:25 +0000909unsigned FLAC__stream_encoder_get_max_residual_partition_order(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000910{
Josh Coalson92031602002-07-24 06:02:11 +0000911 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000912 return encoder->protected_->max_residual_partition_order;
Josh Coalson0a15c142001-06-13 17:59:57 +0000913}
914
Josh Coalson00e53872001-06-16 07:32:25 +0000915unsigned FLAC__stream_encoder_get_rice_parameter_search_dist(const FLAC__StreamEncoder *encoder)
Josh Coalson0a15c142001-06-13 17:59:57 +0000916{
Josh Coalson92031602002-07-24 06:02:11 +0000917 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000918 return encoder->protected_->rice_parameter_search_dist;
Josh Coalson0a15c142001-06-13 17:59:57 +0000919}
920
Josh Coalson3a7b2c92002-08-02 07:38:20 +0000921FLAC__uint64 FLAC__stream_encoder_get_total_samples_estimate(const FLAC__StreamEncoder *encoder)
922{
923 FLAC__ASSERT(0 != encoder);
924 return encoder->protected_->total_samples_estimate;
925}
926
Josh Coalson57ba6f42002-06-07 05:27:37 +0000927FLAC__bool FLAC__stream_encoder_process(FLAC__StreamEncoder *encoder, const FLAC__int32 * const buffer[], unsigned samples)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000928{
929 unsigned i, j, channel;
Josh Coalson77e3f312001-06-23 03:03:24 +0000930 FLAC__int32 x, mid, side;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000931 const unsigned channels = encoder->protected_->channels, blocksize = encoder->protected_->blocksize;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000932
Josh Coalsonf1eff452002-07-31 07:05:33 +0000933 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000934 FLAC__ASSERT(encoder->protected_->state == FLAC__STREAM_ENCODER_OK);
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000935
936 j = 0;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000937 if(encoder->protected_->do_mid_side_stereo && channels == 2) {
Josh Coalsonaa255362001-05-31 06:17:41 +0000938 do {
Josh Coalsond86e03b2002-08-03 21:56:15 +0000939 if(encoder->protected_->verify)
940 append_to_verify_fifo_(&encoder->private_->verify.input_fifo, buffer, j, channels, min(blocksize-encoder->private_->current_sample_number, samples-j));
941
Josh Coalsonfa697a92001-08-16 20:07:29 +0000942 for(i = encoder->private_->current_sample_number; i < blocksize && j < samples; i++, j++) {
Josh Coalson57ba6f42002-06-07 05:27:37 +0000943 x = mid = side = buffer[0][j];
Josh Coalsonfa697a92001-08-16 20:07:29 +0000944 encoder->private_->integer_signal[0][i] = x;
945 encoder->private_->real_signal[0][i] = (FLAC__real)x;
Josh Coalson57ba6f42002-06-07 05:27:37 +0000946 x = buffer[1][j];
Josh Coalsonfa697a92001-08-16 20:07:29 +0000947 encoder->private_->integer_signal[1][i] = x;
948 encoder->private_->real_signal[1][i] = (FLAC__real)x;
Josh Coalsonaa255362001-05-31 06:17:41 +0000949 mid += x;
950 side -= x;
Josh Coalson57ba6f42002-06-07 05:27:37 +0000951 mid >>= 1; /* NOTE: not the same as 'mid = (buffer[0][j] + buffer[1][j]) / 2' ! */
Josh Coalsonfa697a92001-08-16 20:07:29 +0000952 encoder->private_->integer_signal_mid_side[1][i] = side;
953 encoder->private_->integer_signal_mid_side[0][i] = mid;
954 encoder->private_->real_signal_mid_side[1][i] = (FLAC__real)side;
955 encoder->private_->real_signal_mid_side[0][i] = (FLAC__real)mid;
956 encoder->private_->current_sample_number++;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000957 }
Josh Coalsonaa255362001-05-31 06:17:41 +0000958 if(i == blocksize) {
Josh Coalsonf1eff452002-07-31 07:05:33 +0000959 if(!process_frame_(encoder, false)) /* false => not last frame */
Josh Coalsonaa255362001-05-31 06:17:41 +0000960 return false;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000961 }
Josh Coalsonaa255362001-05-31 06:17:41 +0000962 } while(j < samples);
963 }
964 else {
965 do {
Josh Coalsond86e03b2002-08-03 21:56:15 +0000966 if(encoder->protected_->verify)
967 append_to_verify_fifo_(&encoder->private_->verify.input_fifo, buffer, j, channels, min(blocksize-encoder->private_->current_sample_number, samples-j));
968
Josh Coalsonfa697a92001-08-16 20:07:29 +0000969 for(i = encoder->private_->current_sample_number; i < blocksize && j < samples; i++, j++) {
Josh Coalsonaa255362001-05-31 06:17:41 +0000970 for(channel = 0; channel < channels; channel++) {
Josh Coalson57ba6f42002-06-07 05:27:37 +0000971 x = buffer[channel][j];
Josh Coalsonfa697a92001-08-16 20:07:29 +0000972 encoder->private_->integer_signal[channel][i] = x;
973 encoder->private_->real_signal[channel][i] = (FLAC__real)x;
Josh Coalsonaa255362001-05-31 06:17:41 +0000974 }
Josh Coalsonfa697a92001-08-16 20:07:29 +0000975 encoder->private_->current_sample_number++;
Josh Coalsonaa255362001-05-31 06:17:41 +0000976 }
977 if(i == blocksize) {
Josh Coalsonf1eff452002-07-31 07:05:33 +0000978 if(!process_frame_(encoder, false)) /* false => not last frame */
Josh Coalsonaa255362001-05-31 06:17:41 +0000979 return false;
980 }
981 } while(j < samples);
982 }
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000983
984 return true;
985}
986
Josh Coalson57ba6f42002-06-07 05:27:37 +0000987FLAC__bool FLAC__stream_encoder_process_interleaved(FLAC__StreamEncoder *encoder, const FLAC__int32 buffer[], unsigned samples)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000988{
989 unsigned i, j, k, channel;
Josh Coalson77e3f312001-06-23 03:03:24 +0000990 FLAC__int32 x, mid, side;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000991 const unsigned channels = encoder->protected_->channels, blocksize = encoder->protected_->blocksize;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000992
Josh Coalsonf1eff452002-07-31 07:05:33 +0000993 FLAC__ASSERT(0 != encoder);
Josh Coalsonfa697a92001-08-16 20:07:29 +0000994 FLAC__ASSERT(encoder->protected_->state == FLAC__STREAM_ENCODER_OK);
Josh Coalsonbb7f6b92000-12-10 04:09:52 +0000995
996 j = k = 0;
Josh Coalsonfa697a92001-08-16 20:07:29 +0000997 if(encoder->protected_->do_mid_side_stereo && channels == 2) {
Josh Coalsonaa255362001-05-31 06:17:41 +0000998 do {
Josh Coalsond86e03b2002-08-03 21:56:15 +0000999 if(encoder->protected_->verify)
1000 append_to_verify_fifo_interleaved_(&encoder->private_->verify.input_fifo, buffer, j, channels, min(blocksize-encoder->private_->current_sample_number, samples-j));
1001
Josh Coalsonfa697a92001-08-16 20:07:29 +00001002 for(i = encoder->private_->current_sample_number; i < blocksize && j < samples; i++, j++) {
Josh Coalson57ba6f42002-06-07 05:27:37 +00001003 x = mid = side = buffer[k++];
Josh Coalsonfa697a92001-08-16 20:07:29 +00001004 encoder->private_->integer_signal[0][i] = x;
1005 encoder->private_->real_signal[0][i] = (FLAC__real)x;
Josh Coalson57ba6f42002-06-07 05:27:37 +00001006 x = buffer[k++];
Josh Coalsonfa697a92001-08-16 20:07:29 +00001007 encoder->private_->integer_signal[1][i] = x;
1008 encoder->private_->real_signal[1][i] = (FLAC__real)x;
Josh Coalsonaa255362001-05-31 06:17:41 +00001009 mid += x;
1010 side -= x;
1011 mid >>= 1; /* NOTE: not the same as 'mid = (left + right) / 2' ! */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001012 encoder->private_->integer_signal_mid_side[1][i] = side;
1013 encoder->private_->integer_signal_mid_side[0][i] = mid;
1014 encoder->private_->real_signal_mid_side[1][i] = (FLAC__real)side;
1015 encoder->private_->real_signal_mid_side[0][i] = (FLAC__real)mid;
1016 encoder->private_->current_sample_number++;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001017 }
Josh Coalsonaa255362001-05-31 06:17:41 +00001018 if(i == blocksize) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001019 if(!process_frame_(encoder, false)) /* false => not last frame */
Josh Coalsonaa255362001-05-31 06:17:41 +00001020 return false;
1021 }
1022 } while(j < samples);
1023 }
1024 else {
1025 do {
Josh Coalsond86e03b2002-08-03 21:56:15 +00001026 if(encoder->protected_->verify)
1027 append_to_verify_fifo_interleaved_(&encoder->private_->verify.input_fifo, buffer, j, channels, min(blocksize-encoder->private_->current_sample_number, samples-j));
1028
Josh Coalsonfa697a92001-08-16 20:07:29 +00001029 for(i = encoder->private_->current_sample_number; i < blocksize && j < samples; i++, j++) {
Josh Coalsonaa255362001-05-31 06:17:41 +00001030 for(channel = 0; channel < channels; channel++) {
Josh Coalson57ba6f42002-06-07 05:27:37 +00001031 x = buffer[k++];
Josh Coalsonfa697a92001-08-16 20:07:29 +00001032 encoder->private_->integer_signal[channel][i] = x;
1033 encoder->private_->real_signal[channel][i] = (FLAC__real)x;
Josh Coalsonaa255362001-05-31 06:17:41 +00001034 }
Josh Coalsonfa697a92001-08-16 20:07:29 +00001035 encoder->private_->current_sample_number++;
Josh Coalsonaa255362001-05-31 06:17:41 +00001036 }
1037 if(i == blocksize) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001038 if(!process_frame_(encoder, false)) /* false => not last frame */
Josh Coalsonaa255362001-05-31 06:17:41 +00001039 return false;
1040 }
1041 } while(j < samples);
1042 }
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001043
1044 return true;
1045}
1046
Josh Coalsonf1eff452002-07-31 07:05:33 +00001047/***********************************************************************
1048 *
1049 * Private class methods
1050 *
1051 ***********************************************************************/
1052
1053void set_defaults_(FLAC__StreamEncoder *encoder)
Josh Coalson92031602002-07-24 06:02:11 +00001054{
1055 FLAC__ASSERT(0 != encoder);
1056
Josh Coalsond86e03b2002-08-03 21:56:15 +00001057 encoder->protected_->verify = false;
Josh Coalson92031602002-07-24 06:02:11 +00001058 encoder->protected_->streamable_subset = true;
1059 encoder->protected_->do_mid_side_stereo = false;
1060 encoder->protected_->loose_mid_side_stereo = false;
1061 encoder->protected_->channels = 2;
1062 encoder->protected_->bits_per_sample = 16;
1063 encoder->protected_->sample_rate = 44100;
1064 encoder->protected_->blocksize = 1152;
1065 encoder->protected_->max_lpc_order = 0;
1066 encoder->protected_->qlp_coeff_precision = 0;
1067 encoder->protected_->do_qlp_coeff_prec_search = false;
1068 encoder->protected_->do_exhaustive_model_search = false;
1069 encoder->protected_->do_escape_coding = false;
1070 encoder->protected_->min_residual_partition_order = 0;
1071 encoder->protected_->max_residual_partition_order = 0;
1072 encoder->protected_->rice_parameter_search_dist = 0;
1073 encoder->protected_->total_samples_estimate = 0;
1074 encoder->protected_->metadata = 0;
1075 encoder->protected_->num_metadata_blocks = 0;
1076
1077 encoder->private_->write_callback = 0;
1078 encoder->private_->metadata_callback = 0;
1079 encoder->private_->client_data = 0;
1080}
1081
Josh Coalsonf1eff452002-07-31 07:05:33 +00001082void free_(FLAC__StreamEncoder *encoder)
Josh Coalson639aeb02002-07-25 05:38:23 +00001083{
1084 unsigned i, channel;
1085
Josh Coalsonf1eff452002-07-31 07:05:33 +00001086 FLAC__ASSERT(0 != encoder);
Josh Coalson639aeb02002-07-25 05:38:23 +00001087 for(i = 0; i < encoder->protected_->channels; i++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001088 if(0 != encoder->private_->integer_signal_unaligned[i]) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001089 free(encoder->private_->integer_signal_unaligned[i]);
1090 encoder->private_->integer_signal_unaligned[i] = 0;
1091 }
Josh Coalsonf1eff452002-07-31 07:05:33 +00001092 if(0 != encoder->private_->real_signal_unaligned[i]) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001093 free(encoder->private_->real_signal_unaligned[i]);
1094 encoder->private_->real_signal_unaligned[i] = 0;
1095 }
1096 }
1097 for(i = 0; i < 2; i++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001098 if(0 != encoder->private_->integer_signal_mid_side_unaligned[i]) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001099 free(encoder->private_->integer_signal_mid_side_unaligned[i]);
1100 encoder->private_->integer_signal_mid_side_unaligned[i] = 0;
1101 }
Josh Coalsonf1eff452002-07-31 07:05:33 +00001102 if(0 != encoder->private_->real_signal_mid_side_unaligned[i]) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001103 free(encoder->private_->real_signal_mid_side_unaligned[i]);
1104 encoder->private_->real_signal_mid_side_unaligned[i] = 0;
1105 }
1106 }
1107 for(channel = 0; channel < encoder->protected_->channels; channel++) {
1108 for(i = 0; i < 2; i++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001109 if(0 != encoder->private_->residual_workspace_unaligned[channel][i]) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001110 free(encoder->private_->residual_workspace_unaligned[channel][i]);
1111 encoder->private_->residual_workspace_unaligned[channel][i] = 0;
1112 }
1113 }
1114 }
1115 for(channel = 0; channel < 2; channel++) {
1116 for(i = 0; i < 2; i++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001117 if(0 != encoder->private_->residual_workspace_mid_side_unaligned[channel][i]) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001118 free(encoder->private_->residual_workspace_mid_side_unaligned[channel][i]);
1119 encoder->private_->residual_workspace_mid_side_unaligned[channel][i] = 0;
1120 }
1121 }
1122 }
Josh Coalsonf1eff452002-07-31 07:05:33 +00001123 if(0 != encoder->private_->abs_residual_unaligned) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001124 free(encoder->private_->abs_residual_unaligned);
1125 encoder->private_->abs_residual_unaligned = 0;
1126 }
Josh Coalsonf1eff452002-07-31 07:05:33 +00001127 if(0 != encoder->private_->abs_residual_partition_sums_unaligned) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001128 free(encoder->private_->abs_residual_partition_sums_unaligned);
1129 encoder->private_->abs_residual_partition_sums_unaligned = 0;
1130 }
Josh Coalsonf1eff452002-07-31 07:05:33 +00001131 if(0 != encoder->private_->raw_bits_per_partition_unaligned) {
Josh Coalson639aeb02002-07-25 05:38:23 +00001132 free(encoder->private_->raw_bits_per_partition_unaligned);
1133 encoder->private_->raw_bits_per_partition_unaligned = 0;
1134 }
Josh Coalsond86e03b2002-08-03 21:56:15 +00001135 if(encoder->protected_->verify) {
1136 for(i = 0; i < encoder->protected_->channels; i++) {
1137 if(0 != encoder->private_->verify.input_fifo.data[i]) {
1138 free(encoder->private_->verify.input_fifo.data[i]);
1139 encoder->private_->verify.input_fifo.data[i] = 0;
1140 }
1141 }
1142 }
Josh Coalson639aeb02002-07-25 05:38:23 +00001143 FLAC__bitbuffer_free(encoder->private_->frame);
1144}
1145
Josh Coalsonf1eff452002-07-31 07:05:33 +00001146FLAC__bool resize_buffers_(FLAC__StreamEncoder *encoder, unsigned new_size)
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001147{
Josh Coalson77e3f312001-06-23 03:03:24 +00001148 FLAC__bool ok;
Josh Coalson0a15c142001-06-13 17:59:57 +00001149 unsigned i, channel;
1150
1151 FLAC__ASSERT(new_size > 0);
Josh Coalsonfa697a92001-08-16 20:07:29 +00001152 FLAC__ASSERT(encoder->protected_->state == FLAC__STREAM_ENCODER_OK);
1153 FLAC__ASSERT(encoder->private_->current_sample_number == 0);
Josh Coalson0a15c142001-06-13 17:59:57 +00001154
1155 /* To avoid excessive malloc'ing, we only grow the buffer; no shrinking. */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001156 if(new_size <= encoder->private_->input_capacity)
Josh Coalson0a15c142001-06-13 17:59:57 +00001157 return true;
1158
1159 ok = true;
Josh Coalson8395d022001-07-12 21:25:22 +00001160
1161 /* WATCHOUT: FLAC__lpc_compute_residual_from_qlp_coefficients_asm_ia32_mmx() requires that the input arrays (in our case the integer signals) have a buffer of up to 3 zeroes in front (at negative indices) for alignment purposes; we use 4 to keep the data well-aligned. */
1162
Josh Coalsonfa697a92001-08-16 20:07:29 +00001163 for(i = 0; ok && i < encoder->protected_->channels; i++) {
1164 ok = ok && FLAC__memory_alloc_aligned_int32_array(new_size+4, &encoder->private_->integer_signal_unaligned[i], &encoder->private_->integer_signal[i]);
1165 ok = ok && FLAC__memory_alloc_aligned_real_array(new_size, &encoder->private_->real_signal_unaligned[i], &encoder->private_->real_signal[i]);
1166 memset(encoder->private_->integer_signal[i], 0, sizeof(FLAC__int32)*4);
1167 encoder->private_->integer_signal[i] += 4;
Josh Coalson0a15c142001-06-13 17:59:57 +00001168 }
1169 for(i = 0; ok && i < 2; i++) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001170 ok = ok && FLAC__memory_alloc_aligned_int32_array(new_size+4, &encoder->private_->integer_signal_mid_side_unaligned[i], &encoder->private_->integer_signal_mid_side[i]);
1171 ok = ok && FLAC__memory_alloc_aligned_real_array(new_size, &encoder->private_->real_signal_mid_side_unaligned[i], &encoder->private_->real_signal_mid_side[i]);
1172 memset(encoder->private_->integer_signal_mid_side[i], 0, sizeof(FLAC__int32)*4);
1173 encoder->private_->integer_signal_mid_side[i] += 4;
Josh Coalson0a15c142001-06-13 17:59:57 +00001174 }
Josh Coalsonfa697a92001-08-16 20:07:29 +00001175 for(channel = 0; ok && channel < encoder->protected_->channels; channel++) {
Josh Coalson0a15c142001-06-13 17:59:57 +00001176 for(i = 0; ok && i < 2; i++) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001177 ok = ok && FLAC__memory_alloc_aligned_int32_array(new_size, &encoder->private_->residual_workspace_unaligned[channel][i], &encoder->private_->residual_workspace[channel][i]);
Josh Coalson0a15c142001-06-13 17:59:57 +00001178 }
1179 }
1180 for(channel = 0; ok && channel < 2; channel++) {
1181 for(i = 0; ok && i < 2; i++) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001182 ok = ok && FLAC__memory_alloc_aligned_int32_array(new_size, &encoder->private_->residual_workspace_mid_side_unaligned[channel][i], &encoder->private_->residual_workspace_mid_side[channel][i]);
Josh Coalson0a15c142001-06-13 17:59:57 +00001183 }
1184 }
Josh Coalsonfa697a92001-08-16 20:07:29 +00001185 ok = ok && FLAC__memory_alloc_aligned_uint32_array(new_size, &encoder->private_->abs_residual_unaligned, &encoder->private_->abs_residual);
1186 if(encoder->private_->precompute_partition_sums || encoder->protected_->do_escape_coding) /* we require precompute_partition_sums if do_escape_coding because of their intertwined nature */
1187 ok = ok && FLAC__memory_alloc_aligned_uint64_array(new_size * 2, &encoder->private_->abs_residual_partition_sums_unaligned, &encoder->private_->abs_residual_partition_sums);
1188 if(encoder->protected_->do_escape_coding)
1189 ok = ok && FLAC__memory_alloc_aligned_unsigned_array(new_size * 2, &encoder->private_->raw_bits_per_partition_unaligned, &encoder->private_->raw_bits_per_partition);
Josh Coalson0a15c142001-06-13 17:59:57 +00001190
1191 if(ok)
Josh Coalsonfa697a92001-08-16 20:07:29 +00001192 encoder->private_->input_capacity = new_size;
Josh Coalson0a15c142001-06-13 17:59:57 +00001193 else
Josh Coalsonfa697a92001-08-16 20:07:29 +00001194 encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
Josh Coalson0a15c142001-06-13 17:59:57 +00001195
1196 return ok;
1197}
1198
Josh Coalsond86e03b2002-08-03 21:56:15 +00001199FLAC__bool write_bitbuffer_(FLAC__StreamEncoder *encoder, unsigned samples)
Josh Coalson5c491a12002-08-01 06:39:40 +00001200{
1201 const FLAC__byte *buffer;
1202 unsigned bytes;
1203
1204 FLAC__ASSERT(FLAC__bitbuffer_is_byte_aligned(encoder->private_->frame));
1205
1206 FLAC__bitbuffer_get_buffer(encoder->private_->frame, &buffer, &bytes);
1207
Josh Coalsond86e03b2002-08-03 21:56:15 +00001208 if(encoder->protected_->verify) {
1209 encoder->private_->verify.output.data = buffer;
1210 encoder->private_->verify.output.bytes = bytes;
1211 if(encoder->private_->verify.state_hint == ENCODER_IN_MAGIC) {
1212 encoder->private_->verify.needs_magic_hack = true;
1213 }
1214 else {
1215 if(!FLAC__stream_decoder_process_single(encoder->private_->verify.decoder)) {
1216 FLAC__bitbuffer_release_buffer(encoder->private_->frame);
1217 if(encoder->protected_->state != FLAC__STREAM_ENCODER_VERIFY_MISMATCH_IN_AUDIO_DATA)
1218 encoder->protected_->state = FLAC__STREAM_ENCODER_VERIFY_DECODER_ERROR;
1219 return false;
1220 }
1221 }
1222 }
1223
1224 if(encoder->private_->write_callback(encoder, buffer, bytes, samples, encoder->private_->current_frame_number, encoder->private_->client_data) != FLAC__STREAM_ENCODER_WRITE_STATUS_OK) {
1225 encoder->protected_->state = FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_WRITING;
Josh Coalson5c491a12002-08-01 06:39:40 +00001226 return false;
Josh Coalsond86e03b2002-08-03 21:56:15 +00001227 }
Josh Coalson5c491a12002-08-01 06:39:40 +00001228
1229 FLAC__bitbuffer_release_buffer(encoder->private_->frame);
1230
Josh Coalsond86e03b2002-08-03 21:56:15 +00001231 if(samples > 0) {
1232 encoder->private_->metadata.data.stream_info.min_framesize = min(bytes, encoder->private_->metadata.data.stream_info.min_framesize);
1233 encoder->private_->metadata.data.stream_info.max_framesize = max(bytes, encoder->private_->metadata.data.stream_info.max_framesize);
1234 }
1235
Josh Coalson5c491a12002-08-01 06:39:40 +00001236 return true;
1237}
1238
Josh Coalsonf1eff452002-07-31 07:05:33 +00001239FLAC__bool process_frame_(FLAC__StreamEncoder *encoder, FLAC__bool is_last_frame)
Josh Coalson0a15c142001-06-13 17:59:57 +00001240{
Josh Coalsonfa697a92001-08-16 20:07:29 +00001241 FLAC__ASSERT(encoder->protected_->state == FLAC__STREAM_ENCODER_OK);
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001242
1243 /*
Josh Coalsonfa37f1c2001-01-12 23:55:11 +00001244 * Accumulate raw signal to the MD5 signature
1245 */
Josh Coalson57ba6f42002-06-07 05:27:37 +00001246 if(!FLAC__MD5Accumulate(&encoder->private_->md5context, (const FLAC__int32 * const *)encoder->private_->integer_signal, encoder->protected_->channels, encoder->protected_->blocksize, (encoder->protected_->bits_per_sample+7) / 8)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001247 encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
Josh Coalsonfa37f1c2001-01-12 23:55:11 +00001248 return false;
1249 }
1250
1251 /*
Josh Coalson94e02cd2001-01-25 10:41:06 +00001252 * Process the frame header and subframes into the frame bitbuffer
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001253 */
Josh Coalsonf1eff452002-07-31 07:05:33 +00001254 if(!process_subframes_(encoder, is_last_frame)) {
Josh Coalson94e02cd2001-01-25 10:41:06 +00001255 /* the above function sets the state for us in case of an error */
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001256 return false;
1257 }
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001258
1259 /*
1260 * Zero-pad the frame to a byte_boundary
1261 */
Josh Coalsonaec256b2002-03-12 16:19:54 +00001262 if(!FLAC__bitbuffer_zero_pad_to_byte_boundary(encoder->private_->frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001263 encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001264 return false;
1265 }
1266
1267 /*
Josh Coalson215af572001-03-27 01:15:58 +00001268 * CRC-16 the whole thing
1269 */
Josh Coalsonaec256b2002-03-12 16:19:54 +00001270 FLAC__ASSERT(FLAC__bitbuffer_is_byte_aligned(encoder->private_->frame));
1271 FLAC__bitbuffer_write_raw_uint32(encoder->private_->frame, FLAC__bitbuffer_get_write_crc16(encoder->private_->frame), FLAC__FRAME_FOOTER_CRC_LEN);
Josh Coalson215af572001-03-27 01:15:58 +00001272
1273 /*
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001274 * Write it
1275 */
Josh Coalsond86e03b2002-08-03 21:56:15 +00001276 if(!write_bitbuffer_(encoder, encoder->protected_->blocksize)) {
1277 /* the above function sets the state for us in case of an error */
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001278 return false;
1279 }
1280
1281 /*
1282 * Get ready for the next frame
1283 */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001284 encoder->private_->current_sample_number = 0;
1285 encoder->private_->current_frame_number++;
1286 encoder->private_->metadata.data.stream_info.total_samples += (FLAC__uint64)encoder->protected_->blocksize;
Josh Coalsonbb7f6b92000-12-10 04:09:52 +00001287
1288 return true;
1289}
1290
Josh Coalsonf1eff452002-07-31 07:05:33 +00001291FLAC__bool process_subframes_(FLAC__StreamEncoder *encoder, FLAC__bool is_last_frame)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001292{
1293 FLAC__FrameHeader frame_header;
Josh Coalsonfa697a92001-08-16 20:07:29 +00001294 unsigned channel, min_partition_order = encoder->protected_->min_residual_partition_order, max_partition_order;
Josh Coalson8395d022001-07-12 21:25:22 +00001295 FLAC__bool do_independent, do_mid_side, precompute_partition_sums;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001296
1297 /*
Josh Coalson60f77d72001-04-25 02:16:36 +00001298 * Calculate the min,max Rice partition orders
Josh Coalson94e02cd2001-01-25 10:41:06 +00001299 */
1300 if(is_last_frame) {
1301 max_partition_order = 0;
1302 }
1303 else {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001304 unsigned limit = 0, b = encoder->protected_->blocksize;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001305 while(!(b & 1)) {
1306 limit++;
1307 b >>= 1;
1308 }
Josh Coalsonfa697a92001-08-16 20:07:29 +00001309 max_partition_order = min(encoder->protected_->max_residual_partition_order, limit);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001310 }
Josh Coalson60f77d72001-04-25 02:16:36 +00001311 min_partition_order = min(min_partition_order, max_partition_order);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001312
Josh Coalsonfa697a92001-08-16 20:07:29 +00001313 precompute_partition_sums = encoder->private_->precompute_partition_sums && ((max_partition_order > min_partition_order) || encoder->protected_->do_escape_coding);
Josh Coalson8395d022001-07-12 21:25:22 +00001314
Josh Coalson94e02cd2001-01-25 10:41:06 +00001315 /*
1316 * Setup the frame
1317 */
Josh Coalsonaec256b2002-03-12 16:19:54 +00001318 if(!FLAC__bitbuffer_clear(encoder->private_->frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001319 encoder->protected_->state = FLAC__STREAM_ENCODER_MEMORY_ALLOCATION_ERROR;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001320 return false;
1321 }
Josh Coalsonfa697a92001-08-16 20:07:29 +00001322 frame_header.blocksize = encoder->protected_->blocksize;
1323 frame_header.sample_rate = encoder->protected_->sample_rate;
1324 frame_header.channels = encoder->protected_->channels;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001325 frame_header.channel_assignment = FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT; /* the default unless the encoder determines otherwise */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001326 frame_header.bits_per_sample = encoder->protected_->bits_per_sample;
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001327 frame_header.number_type = FLAC__FRAME_NUMBER_TYPE_FRAME_NUMBER;
Josh Coalsonfa697a92001-08-16 20:07:29 +00001328 frame_header.number.frame_number = encoder->private_->current_frame_number;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001329
1330 /*
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001331 * Figure out what channel assignments to try
1332 */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001333 if(encoder->protected_->do_mid_side_stereo) {
1334 if(encoder->protected_->loose_mid_side_stereo) {
1335 if(encoder->private_->loose_mid_side_stereo_frame_count == 0) {
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001336 do_independent = true;
1337 do_mid_side = true;
1338 }
1339 else {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001340 do_independent = (encoder->private_->last_channel_assignment == FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT);
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001341 do_mid_side = !do_independent;
1342 }
1343 }
1344 else {
1345 do_independent = true;
1346 do_mid_side = true;
1347 }
1348 }
1349 else {
1350 do_independent = true;
1351 do_mid_side = false;
1352 }
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001353
Josh Coalson1b689822001-05-31 20:11:02 +00001354 FLAC__ASSERT(do_independent || do_mid_side);
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001355
1356 /*
Josh Coalson82b73242001-03-28 22:17:05 +00001357 * Check for wasted bits; set effective bps for each subframe
Josh Coalson859bc542001-03-27 22:22:27 +00001358 */
1359 if(do_independent) {
Josh Coalson82b73242001-03-28 22:17:05 +00001360 unsigned w;
Josh Coalsonfa697a92001-08-16 20:07:29 +00001361 for(channel = 0; channel < encoder->protected_->channels; channel++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001362 w = get_wasted_bits_(encoder->private_->integer_signal[channel], encoder->protected_->blocksize);
Josh Coalsonfa697a92001-08-16 20:07:29 +00001363 encoder->private_->subframe_workspace[channel][0].wasted_bits = encoder->private_->subframe_workspace[channel][1].wasted_bits = w;
1364 encoder->private_->subframe_bps[channel] = encoder->protected_->bits_per_sample - w;
Josh Coalson82b73242001-03-28 22:17:05 +00001365 }
Josh Coalson859bc542001-03-27 22:22:27 +00001366 }
1367 if(do_mid_side) {
Josh Coalson82b73242001-03-28 22:17:05 +00001368 unsigned w;
Josh Coalsonfa697a92001-08-16 20:07:29 +00001369 FLAC__ASSERT(encoder->protected_->channels == 2);
Josh Coalson82b73242001-03-28 22:17:05 +00001370 for(channel = 0; channel < 2; channel++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001371 w = get_wasted_bits_(encoder->private_->integer_signal_mid_side[channel], encoder->protected_->blocksize);
Josh Coalsonfa697a92001-08-16 20:07:29 +00001372 encoder->private_->subframe_workspace_mid_side[channel][0].wasted_bits = encoder->private_->subframe_workspace_mid_side[channel][1].wasted_bits = w;
1373 encoder->private_->subframe_bps_mid_side[channel] = encoder->protected_->bits_per_sample - w + (channel==0? 0:1);
Josh Coalson82b73242001-03-28 22:17:05 +00001374 }
Josh Coalson859bc542001-03-27 22:22:27 +00001375 }
1376
1377 /*
Josh Coalson94e02cd2001-01-25 10:41:06 +00001378 * First do a normal encoding pass of each independent channel
1379 */
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001380 if(do_independent) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001381 for(channel = 0; channel < encoder->protected_->channels; channel++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001382 if(!process_subframe_(encoder, min_partition_order, max_partition_order, precompute_partition_sums, false, &frame_header, encoder->private_->subframe_bps[channel], encoder->private_->integer_signal[channel], encoder->private_->real_signal[channel], encoder->private_->subframe_workspace_ptr[channel], encoder->private_->residual_workspace[channel], encoder->private_->best_subframe+channel, encoder->private_->best_subframe_bits+channel))
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001383 return false;
1384 }
Josh Coalson94e02cd2001-01-25 10:41:06 +00001385 }
1386
1387 /*
1388 * Now do mid and side channels if requested
1389 */
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001390 if(do_mid_side) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001391 FLAC__ASSERT(encoder->protected_->channels == 2);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001392
1393 for(channel = 0; channel < 2; channel++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001394 if(!process_subframe_(encoder, min_partition_order, max_partition_order, precompute_partition_sums, false, &frame_header, encoder->private_->subframe_bps_mid_side[channel], encoder->private_->integer_signal_mid_side[channel], encoder->private_->real_signal_mid_side[channel], encoder->private_->subframe_workspace_ptr_mid_side[channel], encoder->private_->residual_workspace_mid_side[channel], encoder->private_->best_subframe_mid_side+channel, encoder->private_->best_subframe_bits_mid_side+channel))
Josh Coalson94e02cd2001-01-25 10:41:06 +00001395 return false;
1396 }
1397 }
1398
1399 /*
1400 * Compose the frame bitbuffer
1401 */
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001402 if(do_mid_side) {
Josh Coalson82b73242001-03-28 22:17:05 +00001403 unsigned left_bps = 0, right_bps = 0; /* initialized only to prevent superfluous compiler warning */
1404 FLAC__Subframe *left_subframe = 0, *right_subframe = 0; /* initialized only to prevent superfluous compiler warning */
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001405 FLAC__ChannelAssignment channel_assignment;
1406
Josh Coalsonfa697a92001-08-16 20:07:29 +00001407 FLAC__ASSERT(encoder->protected_->channels == 2);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001408
Josh Coalsonfa697a92001-08-16 20:07:29 +00001409 if(encoder->protected_->loose_mid_side_stereo && encoder->private_->loose_mid_side_stereo_frame_count > 0) {
1410 channel_assignment = (encoder->private_->last_channel_assignment == FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT? FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT : FLAC__CHANNEL_ASSIGNMENT_MID_SIDE);
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001411 }
1412 else {
1413 unsigned bits[4]; /* WATCHOUT - indexed by FLAC__ChannelAssignment */
1414 unsigned min_bits;
1415 FLAC__ChannelAssignment ca;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001416
Josh Coalson1b689822001-05-31 20:11:02 +00001417 FLAC__ASSERT(do_independent && do_mid_side);
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001418
1419 /* We have to figure out which channel assignent results in the smallest frame */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001420 bits[FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT] = encoder->private_->best_subframe_bits [0] + encoder->private_->best_subframe_bits [1];
1421 bits[FLAC__CHANNEL_ASSIGNMENT_LEFT_SIDE ] = encoder->private_->best_subframe_bits [0] + encoder->private_->best_subframe_bits_mid_side[1];
1422 bits[FLAC__CHANNEL_ASSIGNMENT_RIGHT_SIDE ] = encoder->private_->best_subframe_bits [1] + encoder->private_->best_subframe_bits_mid_side[1];
1423 bits[FLAC__CHANNEL_ASSIGNMENT_MID_SIDE ] = encoder->private_->best_subframe_bits_mid_side[0] + encoder->private_->best_subframe_bits_mid_side[1];
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001424
1425 for(channel_assignment = 0, min_bits = bits[0], ca = 1; ca <= 3; ca++) {
1426 if(bits[ca] < min_bits) {
1427 min_bits = bits[ca];
1428 channel_assignment = ca;
1429 }
Josh Coalson94e02cd2001-01-25 10:41:06 +00001430 }
1431 }
1432
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001433 frame_header.channel_assignment = channel_assignment;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001434
Josh Coalsonaec256b2002-03-12 16:19:54 +00001435 if(!FLAC__frame_add_header(&frame_header, encoder->protected_->streamable_subset, is_last_frame, encoder->private_->frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001436 encoder->protected_->state = FLAC__STREAM_ENCODER_FRAMING_ERROR;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001437 return false;
1438 }
1439
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001440 switch(channel_assignment) {
Josh Coalson94e02cd2001-01-25 10:41:06 +00001441 case FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001442 left_subframe = &encoder->private_->subframe_workspace [0][encoder->private_->best_subframe [0]];
1443 right_subframe = &encoder->private_->subframe_workspace [1][encoder->private_->best_subframe [1]];
Josh Coalson94e02cd2001-01-25 10:41:06 +00001444 break;
1445 case FLAC__CHANNEL_ASSIGNMENT_LEFT_SIDE:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001446 left_subframe = &encoder->private_->subframe_workspace [0][encoder->private_->best_subframe [0]];
1447 right_subframe = &encoder->private_->subframe_workspace_mid_side[1][encoder->private_->best_subframe_mid_side[1]];
Josh Coalson94e02cd2001-01-25 10:41:06 +00001448 break;
1449 case FLAC__CHANNEL_ASSIGNMENT_RIGHT_SIDE:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001450 left_subframe = &encoder->private_->subframe_workspace_mid_side[1][encoder->private_->best_subframe_mid_side[1]];
1451 right_subframe = &encoder->private_->subframe_workspace [1][encoder->private_->best_subframe [1]];
Josh Coalson94e02cd2001-01-25 10:41:06 +00001452 break;
1453 case FLAC__CHANNEL_ASSIGNMENT_MID_SIDE:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001454 left_subframe = &encoder->private_->subframe_workspace_mid_side[0][encoder->private_->best_subframe_mid_side[0]];
1455 right_subframe = &encoder->private_->subframe_workspace_mid_side[1][encoder->private_->best_subframe_mid_side[1]];
Josh Coalson94e02cd2001-01-25 10:41:06 +00001456 break;
1457 default:
Josh Coalson1b689822001-05-31 20:11:02 +00001458 FLAC__ASSERT(0);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001459 }
Josh Coalson82b73242001-03-28 22:17:05 +00001460
1461 switch(channel_assignment) {
1462 case FLAC__CHANNEL_ASSIGNMENT_INDEPENDENT:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001463 left_bps = encoder->private_->subframe_bps [0];
1464 right_bps = encoder->private_->subframe_bps [1];
Josh Coalson82b73242001-03-28 22:17:05 +00001465 break;
1466 case FLAC__CHANNEL_ASSIGNMENT_LEFT_SIDE:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001467 left_bps = encoder->private_->subframe_bps [0];
1468 right_bps = encoder->private_->subframe_bps_mid_side[1];
Josh Coalson82b73242001-03-28 22:17:05 +00001469 break;
1470 case FLAC__CHANNEL_ASSIGNMENT_RIGHT_SIDE:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001471 left_bps = encoder->private_->subframe_bps_mid_side[1];
1472 right_bps = encoder->private_->subframe_bps [1];
Josh Coalson82b73242001-03-28 22:17:05 +00001473 break;
1474 case FLAC__CHANNEL_ASSIGNMENT_MID_SIDE:
Josh Coalsonfa697a92001-08-16 20:07:29 +00001475 left_bps = encoder->private_->subframe_bps_mid_side[0];
1476 right_bps = encoder->private_->subframe_bps_mid_side[1];
Josh Coalson82b73242001-03-28 22:17:05 +00001477 break;
1478 default:
Josh Coalson1b689822001-05-31 20:11:02 +00001479 FLAC__ASSERT(0);
Josh Coalson82b73242001-03-28 22:17:05 +00001480 }
1481
1482 /* note that encoder_add_subframe_ sets the state for us in case of an error */
Josh Coalsonf1eff452002-07-31 07:05:33 +00001483 if(!add_subframe_(encoder, &frame_header, left_bps , left_subframe , encoder->private_->frame))
Josh Coalson82b73242001-03-28 22:17:05 +00001484 return false;
Josh Coalsonf1eff452002-07-31 07:05:33 +00001485 if(!add_subframe_(encoder, &frame_header, right_bps, right_subframe, encoder->private_->frame))
Josh Coalson82b73242001-03-28 22:17:05 +00001486 return false;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001487 }
1488 else {
Josh Coalsonaec256b2002-03-12 16:19:54 +00001489 if(!FLAC__frame_add_header(&frame_header, encoder->protected_->streamable_subset, is_last_frame, encoder->private_->frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001490 encoder->protected_->state = FLAC__STREAM_ENCODER_FRAMING_ERROR;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001491 return false;
1492 }
1493
Josh Coalsonfa697a92001-08-16 20:07:29 +00001494 for(channel = 0; channel < encoder->protected_->channels; channel++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001495 if(!add_subframe_(encoder, &frame_header, encoder->private_->subframe_bps[channel], &encoder->private_->subframe_workspace[channel][encoder->private_->best_subframe[channel]], encoder->private_->frame)) {
Josh Coalson94e02cd2001-01-25 10:41:06 +00001496 /* the above function sets the state for us in case of an error */
1497 return false;
1498 }
1499 }
1500 }
1501
Josh Coalsonfa697a92001-08-16 20:07:29 +00001502 if(encoder->protected_->loose_mid_side_stereo) {
1503 encoder->private_->loose_mid_side_stereo_frame_count++;
1504 if(encoder->private_->loose_mid_side_stereo_frame_count >= encoder->private_->loose_mid_side_stereo_frames)
1505 encoder->private_->loose_mid_side_stereo_frame_count = 0;
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001506 }
1507
Josh Coalsonfa697a92001-08-16 20:07:29 +00001508 encoder->private_->last_channel_assignment = frame_header.channel_assignment;
Josh Coalsonb5e60e52001-01-28 09:27:27 +00001509
Josh Coalson94e02cd2001-01-25 10:41:06 +00001510 return true;
1511}
1512
Josh Coalsonf1eff452002-07-31 07:05:33 +00001513FLAC__bool process_subframe_(FLAC__StreamEncoder *encoder, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool verbatim_only, const FLAC__FrameHeader *frame_header, unsigned subframe_bps, const FLAC__int32 integer_signal[], const FLAC__real real_signal[], FLAC__Subframe *subframe[2], FLAC__int32 *residual[2], unsigned *best_subframe, unsigned *best_bits)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001514{
Josh Coalson77e3f312001-06-23 03:03:24 +00001515 FLAC__real fixed_residual_bits_per_sample[FLAC__MAX_FIXED_ORDER+1];
1516 FLAC__real lpc_residual_bits_per_sample;
Josh Coalsonfa697a92001-08-16 20:07:29 +00001517 FLAC__real autoc[FLAC__MAX_LPC_ORDER+1]; /* WATCHOUT: the size is important even though encoder->protected_->max_lpc_order might be less; some asm routines need all the space */
Josh Coalson77e3f312001-06-23 03:03:24 +00001518 FLAC__real lpc_error[FLAC__MAX_LPC_ORDER];
Josh Coalson94e02cd2001-01-25 10:41:06 +00001519 unsigned min_lpc_order, max_lpc_order, lpc_order;
1520 unsigned min_fixed_order, max_fixed_order, guess_fixed_order, fixed_order;
1521 unsigned min_qlp_coeff_precision, max_qlp_coeff_precision, qlp_coeff_precision;
1522 unsigned rice_parameter;
1523 unsigned _candidate_bits, _best_bits;
1524 unsigned _best_subframe;
1525
1526 /* verbatim subframe is the baseline against which we measure other compressed subframes */
1527 _best_subframe = 0;
Josh Coalsonf1eff452002-07-31 07:05:33 +00001528 _best_bits = evaluate_verbatim_subframe_(integer_signal, frame_header->blocksize, subframe_bps, subframe[_best_subframe]);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001529
1530 if(!verbatim_only && frame_header->blocksize >= FLAC__MAX_FIXED_ORDER) {
1531 /* check for constant subframe */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001532 guess_fixed_order = encoder->private_->local_fixed_compute_best_predictor(integer_signal+FLAC__MAX_FIXED_ORDER, frame_header->blocksize-FLAC__MAX_FIXED_ORDER, fixed_residual_bits_per_sample);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001533 if(fixed_residual_bits_per_sample[1] == 0.0) {
1534 /* the above means integer_signal+FLAC__MAX_FIXED_ORDER is constant, now we just have to check the warmup samples */
1535 unsigned i, signal_is_constant = true;
1536 for(i = 1; i <= FLAC__MAX_FIXED_ORDER; i++) {
1537 if(integer_signal[0] != integer_signal[i]) {
1538 signal_is_constant = false;
1539 break;
1540 }
1541 }
1542 if(signal_is_constant) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001543 _candidate_bits = evaluate_constant_subframe_(integer_signal[0], subframe_bps, subframe[!_best_subframe]);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001544 if(_candidate_bits < _best_bits) {
1545 _best_subframe = !_best_subframe;
1546 _best_bits = _candidate_bits;
1547 }
1548 }
1549 }
1550 else {
1551 /* encode fixed */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001552 if(encoder->protected_->do_exhaustive_model_search) {
Josh Coalson94e02cd2001-01-25 10:41:06 +00001553 min_fixed_order = 0;
1554 max_fixed_order = FLAC__MAX_FIXED_ORDER;
1555 }
1556 else {
1557 min_fixed_order = max_fixed_order = guess_fixed_order;
1558 }
1559 for(fixed_order = min_fixed_order; fixed_order <= max_fixed_order; fixed_order++) {
Josh Coalson77e3f312001-06-23 03:03:24 +00001560 if(fixed_residual_bits_per_sample[fixed_order] >= (FLAC__real)subframe_bps)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001561 continue; /* don't even try */
Josh Coalson46f2ae82001-02-08 00:27:21 +00001562 rice_parameter = (fixed_residual_bits_per_sample[fixed_order] > 0.0)? (unsigned)(fixed_residual_bits_per_sample[fixed_order]+0.5) : 0; /* 0.5 is for rounding */
Josh Coalsonbb6712e2001-04-24 22:54:07 +00001563#ifndef FLAC__SYMMETRIC_RICE
Josh Coalson46f2ae82001-02-08 00:27:21 +00001564 rice_parameter++; /* to account for the signed->unsigned conversion during rice coding */
Josh Coalsonb9433f92001-03-17 01:07:00 +00001565#endif
Josh Coalson8395d022001-07-12 21:25:22 +00001566 if(rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00001567#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00001568 fprintf(stderr, "clipping rice_parameter (%u -> %u) @0\n", rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
1569#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00001570 rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
Josh Coalson8395d022001-07-12 21:25:22 +00001571 }
Josh Coalsonf1eff452002-07-31 07:05:33 +00001572 _candidate_bits = evaluate_fixed_subframe_(encoder, integer_signal, residual[!_best_subframe], encoder->private_->abs_residual, encoder->private_->abs_residual_partition_sums, encoder->private_->raw_bits_per_partition, frame_header->blocksize, subframe_bps, fixed_order, rice_parameter, min_partition_order, max_partition_order, precompute_partition_sums, encoder->protected_->do_escape_coding, encoder->protected_->rice_parameter_search_dist, subframe[!_best_subframe]);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001573 if(_candidate_bits < _best_bits) {
1574 _best_subframe = !_best_subframe;
1575 _best_bits = _candidate_bits;
1576 }
1577 }
1578
1579 /* encode lpc */
Josh Coalsonfa697a92001-08-16 20:07:29 +00001580 if(encoder->protected_->max_lpc_order > 0) {
1581 if(encoder->protected_->max_lpc_order >= frame_header->blocksize)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001582 max_lpc_order = frame_header->blocksize-1;
1583 else
Josh Coalsonfa697a92001-08-16 20:07:29 +00001584 max_lpc_order = encoder->protected_->max_lpc_order;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001585 if(max_lpc_order > 0) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001586 encoder->private_->local_lpc_compute_autocorrelation(real_signal, frame_header->blocksize, max_lpc_order+1, autoc);
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001587 /* if autoc[0] == 0.0, the signal is constant and we usually won't get here, but it can happen */
1588 if(autoc[0] != 0.0) {
Josh Coalson8084b052001-11-01 00:27:29 +00001589 FLAC__lpc_compute_lp_coefficients(autoc, max_lpc_order, encoder->private_->lp_coeff, lpc_error);
Josh Coalsonfa697a92001-08-16 20:07:29 +00001590 if(encoder->protected_->do_exhaustive_model_search) {
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001591 min_lpc_order = 1;
1592 }
1593 else {
Josh Coalson82b73242001-03-28 22:17:05 +00001594 unsigned guess_lpc_order = FLAC__lpc_compute_best_order(lpc_error, max_lpc_order, frame_header->blocksize, subframe_bps);
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001595 min_lpc_order = max_lpc_order = guess_lpc_order;
1596 }
Josh Coalsonfa697a92001-08-16 20:07:29 +00001597 if(encoder->protected_->do_qlp_coeff_prec_search) {
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001598 min_qlp_coeff_precision = FLAC__MIN_QLP_COEFF_PRECISION;
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001599 max_qlp_coeff_precision = min(8*sizeof(FLAC__int32) - subframe_bps - 1 - 2, (1u<<FLAC__SUBFRAME_LPC_QLP_COEFF_PRECISION_LEN)-1); /* -2 to keep things 32-bit safe */
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001600 }
1601 else {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001602 min_qlp_coeff_precision = max_qlp_coeff_precision = encoder->protected_->qlp_coeff_precision;
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001603 }
1604 for(lpc_order = min_lpc_order; lpc_order <= max_lpc_order; lpc_order++) {
1605 lpc_residual_bits_per_sample = FLAC__lpc_compute_expected_bits_per_residual_sample(lpc_error[lpc_order-1], frame_header->blocksize-lpc_order);
Josh Coalson77e3f312001-06-23 03:03:24 +00001606 if(lpc_residual_bits_per_sample >= (FLAC__real)subframe_bps)
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001607 continue; /* don't even try */
1608 rice_parameter = (lpc_residual_bits_per_sample > 0.0)? (unsigned)(lpc_residual_bits_per_sample+0.5) : 0; /* 0.5 is for rounding */
Josh Coalsonbb6712e2001-04-24 22:54:07 +00001609#ifndef FLAC__SYMMETRIC_RICE
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001610 rice_parameter++; /* to account for the signed->unsigned conversion during rice coding */
Josh Coalsonb9433f92001-03-17 01:07:00 +00001611#endif
Josh Coalson8395d022001-07-12 21:25:22 +00001612 if(rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00001613#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00001614 fprintf(stderr, "clipping rice_parameter (%u -> %u) @1\n", rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
1615#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00001616 rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
Josh Coalson8395d022001-07-12 21:25:22 +00001617 }
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001618 for(qlp_coeff_precision = min_qlp_coeff_precision; qlp_coeff_precision <= max_qlp_coeff_precision; qlp_coeff_precision++) {
Josh Coalsonf1eff452002-07-31 07:05:33 +00001619 _candidate_bits = evaluate_lpc_subframe_(encoder, integer_signal, residual[!_best_subframe], encoder->private_->abs_residual, encoder->private_->abs_residual_partition_sums, encoder->private_->raw_bits_per_partition, encoder->private_->lp_coeff[lpc_order-1], frame_header->blocksize, subframe_bps, lpc_order, qlp_coeff_precision, rice_parameter, min_partition_order, max_partition_order, precompute_partition_sums, encoder->protected_->do_escape_coding, encoder->protected_->rice_parameter_search_dist, subframe[!_best_subframe]);
Josh Coalsonf4ce50b2001-02-28 23:45:15 +00001620 if(_candidate_bits > 0) { /* if == 0, there was a problem quantizing the lpcoeffs */
1621 if(_candidate_bits < _best_bits) {
1622 _best_subframe = !_best_subframe;
1623 _best_bits = _candidate_bits;
1624 }
Josh Coalson94e02cd2001-01-25 10:41:06 +00001625 }
1626 }
1627 }
1628 }
1629 }
1630 }
1631 }
1632 }
1633
1634 *best_subframe = _best_subframe;
1635 *best_bits = _best_bits;
1636
1637 return true;
1638}
1639
Josh Coalsonf1eff452002-07-31 07:05:33 +00001640FLAC__bool add_subframe_(FLAC__StreamEncoder *encoder, const FLAC__FrameHeader *frame_header, unsigned subframe_bps, const FLAC__Subframe *subframe, FLAC__BitBuffer *frame)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001641{
1642 switch(subframe->type) {
1643 case FLAC__SUBFRAME_TYPE_CONSTANT:
Josh Coalson82b73242001-03-28 22:17:05 +00001644 if(!FLAC__subframe_add_constant(&(subframe->data.constant), subframe_bps, subframe->wasted_bits, frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001645 encoder->protected_->state = FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_ENCODING;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001646 return false;
1647 }
1648 break;
1649 case FLAC__SUBFRAME_TYPE_FIXED:
Josh Coalson82b73242001-03-28 22:17:05 +00001650 if(!FLAC__subframe_add_fixed(&(subframe->data.fixed), frame_header->blocksize - subframe->data.fixed.order, subframe_bps, subframe->wasted_bits, frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001651 encoder->protected_->state = FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_ENCODING;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001652 return false;
1653 }
1654 break;
1655 case FLAC__SUBFRAME_TYPE_LPC:
Josh Coalson82b73242001-03-28 22:17:05 +00001656 if(!FLAC__subframe_add_lpc(&(subframe->data.lpc), frame_header->blocksize - subframe->data.lpc.order, subframe_bps, subframe->wasted_bits, frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001657 encoder->protected_->state = FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_ENCODING;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001658 return false;
1659 }
1660 break;
1661 case FLAC__SUBFRAME_TYPE_VERBATIM:
Josh Coalson82b73242001-03-28 22:17:05 +00001662 if(!FLAC__subframe_add_verbatim(&(subframe->data.verbatim), frame_header->blocksize, subframe_bps, subframe->wasted_bits, frame)) {
Josh Coalsonfa697a92001-08-16 20:07:29 +00001663 encoder->protected_->state = FLAC__STREAM_ENCODER_FATAL_ERROR_WHILE_ENCODING;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001664 return false;
1665 }
1666 break;
1667 default:
Josh Coalson1b689822001-05-31 20:11:02 +00001668 FLAC__ASSERT(0);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001669 }
1670
1671 return true;
1672}
1673
Josh Coalsonf1eff452002-07-31 07:05:33 +00001674unsigned evaluate_constant_subframe_(const FLAC__int32 signal, unsigned subframe_bps, FLAC__Subframe *subframe)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001675{
1676 subframe->type = FLAC__SUBFRAME_TYPE_CONSTANT;
1677 subframe->data.constant.value = signal;
1678
Josh Coalson82b73242001-03-28 22:17:05 +00001679 return FLAC__SUBFRAME_ZERO_PAD_LEN + FLAC__SUBFRAME_TYPE_LEN + FLAC__SUBFRAME_WASTED_BITS_FLAG_LEN + subframe_bps;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001680}
1681
Josh Coalsonf1eff452002-07-31 07:05:33 +00001682unsigned evaluate_fixed_subframe_(FLAC__StreamEncoder *encoder, const FLAC__int32 signal[], FLAC__int32 residual[], FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned raw_bits_per_partition[], unsigned blocksize, unsigned subframe_bps, unsigned order, unsigned rice_parameter, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool do_escape_coding, unsigned rice_parameter_search_dist, FLAC__Subframe *subframe)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001683{
1684 unsigned i, residual_bits;
1685 const unsigned residual_samples = blocksize - order;
1686
1687 FLAC__fixed_compute_residual(signal+order, residual_samples, order, residual);
1688
1689 subframe->type = FLAC__SUBFRAME_TYPE_FIXED;
1690
1691 subframe->data.fixed.entropy_coding_method.type = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE;
1692 subframe->data.fixed.residual = residual;
1693
Josh Coalsonf1eff452002-07-31 07:05:33 +00001694 residual_bits = find_best_partition_order_(encoder->private_, residual, abs_residual, abs_residual_partition_sums, raw_bits_per_partition, residual_samples, order, rice_parameter, min_partition_order, max_partition_order, precompute_partition_sums, do_escape_coding, rice_parameter_search_dist, &subframe->data.fixed.entropy_coding_method.data.partitioned_rice.order, subframe->data.fixed.entropy_coding_method.data.partitioned_rice.parameters, subframe->data.fixed.entropy_coding_method.data.partitioned_rice.raw_bits);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001695
1696 subframe->data.fixed.order = order;
1697 for(i = 0; i < order; i++)
1698 subframe->data.fixed.warmup[i] = signal[i];
1699
Josh Coalson82b73242001-03-28 22:17:05 +00001700 return FLAC__SUBFRAME_ZERO_PAD_LEN + FLAC__SUBFRAME_TYPE_LEN + FLAC__SUBFRAME_WASTED_BITS_FLAG_LEN + (order * subframe_bps) + residual_bits;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001701}
1702
Josh Coalsonf1eff452002-07-31 07:05:33 +00001703unsigned evaluate_lpc_subframe_(FLAC__StreamEncoder *encoder, const FLAC__int32 signal[], FLAC__int32 residual[], FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned raw_bits_per_partition[], const FLAC__real lp_coeff[], unsigned blocksize, unsigned subframe_bps, unsigned order, unsigned qlp_coeff_precision, unsigned rice_parameter, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool do_escape_coding, unsigned rice_parameter_search_dist, FLAC__Subframe *subframe)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001704{
Josh Coalson77e3f312001-06-23 03:03:24 +00001705 FLAC__int32 qlp_coeff[FLAC__MAX_LPC_ORDER];
Josh Coalson94e02cd2001-01-25 10:41:06 +00001706 unsigned i, residual_bits;
1707 int quantization, ret;
1708 const unsigned residual_samples = blocksize - order;
1709
Josh Coalson82b73242001-03-28 22:17:05 +00001710 ret = FLAC__lpc_quantize_coefficients(lp_coeff, order, qlp_coeff_precision, subframe_bps, qlp_coeff, &quantization);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001711 if(ret != 0)
1712 return 0; /* this is a hack to indicate to the caller that we can't do lp at this order on this subframe */
1713
Josh Coalson92d42402001-05-31 20:53:19 +00001714 if(subframe_bps <= 16 && qlp_coeff_precision <= 16)
Josh Coalsonfa697a92001-08-16 20:07:29 +00001715 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients_16bit(signal+order, residual_samples, qlp_coeff, order, quantization, residual);
Josh Coalson92d42402001-05-31 20:53:19 +00001716 else
Josh Coalsonfa697a92001-08-16 20:07:29 +00001717 encoder->private_->local_lpc_compute_residual_from_qlp_coefficients(signal+order, residual_samples, qlp_coeff, order, quantization, residual);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001718
1719 subframe->type = FLAC__SUBFRAME_TYPE_LPC;
1720
1721 subframe->data.lpc.entropy_coding_method.type = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE;
1722 subframe->data.lpc.residual = residual;
1723
Josh Coalsonf1eff452002-07-31 07:05:33 +00001724 residual_bits = find_best_partition_order_(encoder->private_, residual, abs_residual, abs_residual_partition_sums, raw_bits_per_partition, residual_samples, order, rice_parameter, min_partition_order, max_partition_order, precompute_partition_sums, do_escape_coding, rice_parameter_search_dist, &subframe->data.lpc.entropy_coding_method.data.partitioned_rice.order, subframe->data.lpc.entropy_coding_method.data.partitioned_rice.parameters, subframe->data.lpc.entropy_coding_method.data.partitioned_rice.raw_bits);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001725
1726 subframe->data.lpc.order = order;
1727 subframe->data.lpc.qlp_coeff_precision = qlp_coeff_precision;
1728 subframe->data.lpc.quantization_level = quantization;
Josh Coalson77e3f312001-06-23 03:03:24 +00001729 memcpy(subframe->data.lpc.qlp_coeff, qlp_coeff, sizeof(FLAC__int32)*FLAC__MAX_LPC_ORDER);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001730 for(i = 0; i < order; i++)
1731 subframe->data.lpc.warmup[i] = signal[i];
1732
Josh Coalson82b73242001-03-28 22:17:05 +00001733 return FLAC__SUBFRAME_ZERO_PAD_LEN + FLAC__SUBFRAME_TYPE_LEN + FLAC__SUBFRAME_WASTED_BITS_FLAG_LEN + FLAC__SUBFRAME_LPC_QLP_COEFF_PRECISION_LEN + FLAC__SUBFRAME_LPC_QLP_SHIFT_LEN + (order * (qlp_coeff_precision + subframe_bps)) + residual_bits;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001734}
1735
Josh Coalsonf1eff452002-07-31 07:05:33 +00001736unsigned evaluate_verbatim_subframe_(const FLAC__int32 signal[], unsigned blocksize, unsigned subframe_bps, FLAC__Subframe *subframe)
Josh Coalson94e02cd2001-01-25 10:41:06 +00001737{
1738 subframe->type = FLAC__SUBFRAME_TYPE_VERBATIM;
1739
1740 subframe->data.verbatim.data = signal;
1741
Josh Coalson82b73242001-03-28 22:17:05 +00001742 return FLAC__SUBFRAME_ZERO_PAD_LEN + FLAC__SUBFRAME_TYPE_LEN + FLAC__SUBFRAME_WASTED_BITS_FLAG_LEN + (blocksize * subframe_bps);
Josh Coalson94e02cd2001-01-25 10:41:06 +00001743}
1744
Josh Coalsonf1eff452002-07-31 07:05:33 +00001745unsigned find_best_partition_order_(FLAC__StreamEncoderPrivate *private_, const FLAC__int32 residual[], FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned raw_bits_per_partition[], unsigned residual_samples, unsigned predictor_order, unsigned rice_parameter, unsigned min_partition_order, unsigned max_partition_order, FLAC__bool precompute_partition_sums, FLAC__bool do_escape_coding, unsigned rice_parameter_search_dist, unsigned *best_partition_order, unsigned best_parameters[], unsigned best_raw_bits[])
Josh Coalson94e02cd2001-01-25 10:41:06 +00001746{
Josh Coalson77e3f312001-06-23 03:03:24 +00001747 FLAC__int32 r;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001748 unsigned residual_bits, best_residual_bits = 0;
Josh Coalsonafcd8772001-04-18 22:59:25 +00001749 unsigned residual_sample;
Josh Coalson8084b052001-11-01 00:27:29 +00001750 unsigned best_parameters_index = 0;
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001751 const unsigned blocksize = residual_samples + predictor_order;
Josh Coalson94e02cd2001-01-25 10:41:06 +00001752
Josh Coalson2051dd42001-04-12 22:22:34 +00001753 /* compute abs(residual) for use later */
1754 for(residual_sample = 0; residual_sample < residual_samples; residual_sample++) {
1755 r = residual[residual_sample];
Josh Coalson77e3f312001-06-23 03:03:24 +00001756 abs_residual[residual_sample] = (FLAC__uint32)(r<0? -r : r);
Josh Coalson2051dd42001-04-12 22:22:34 +00001757 }
1758
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001759 while(max_partition_order > 0 && blocksize >> max_partition_order <= predictor_order)
1760 max_partition_order--;
1761 FLAC__ASSERT(blocksize >> max_partition_order > predictor_order);
1762 min_partition_order = min(min_partition_order, max_partition_order);
1763
Josh Coalson8395d022001-07-12 21:25:22 +00001764 if(precompute_partition_sums) {
1765 int partition_order;
1766 unsigned sum;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001767
Josh Coalsonf1eff452002-07-31 07:05:33 +00001768 precompute_partition_info_sums_(abs_residual, abs_residual_partition_sums, residual_samples, predictor_order, min_partition_order, max_partition_order);
Josh Coalson8395d022001-07-12 21:25:22 +00001769
1770 if(do_escape_coding)
Josh Coalsonf1eff452002-07-31 07:05:33 +00001771 precompute_partition_info_escapes_(residual, raw_bits_per_partition, residual_samples, predictor_order, min_partition_order, max_partition_order);
Josh Coalson8395d022001-07-12 21:25:22 +00001772
1773 for(partition_order = (int)max_partition_order, sum = 0; partition_order >= (int)min_partition_order; partition_order--) {
1774#ifdef DONT_ESTIMATE_RICE_BITS
Josh Coalsonf1eff452002-07-31 07:05:33 +00001775 if(!set_partitioned_rice_with_precompute_(residual, abs_residual_partition_sums+sum, raw_bits_per_partition+sum, residual_samples, predictor_order, rice_parameter, rice_parameter_search_dist, (unsigned)partition_order, do_escape_coding, private_->parameters[!best_parameters_index], private_->raw_bits[!best_parameters_index], &residual_bits))
Josh Coalsonafcd8772001-04-18 22:59:25 +00001776#else
Josh Coalsonf1eff452002-07-31 07:05:33 +00001777 if(!set_partitioned_rice_with_precompute_(abs_residual, abs_residual_partition_sums+sum, raw_bits_per_partition+sum, residual_samples, predictor_order, rice_parameter, rice_parameter_search_dist, (unsigned)partition_order, do_escape_coding, private_->parameters[!best_parameters_index], private_->raw_bits[!best_parameters_index], &residual_bits))
Josh Coalson8395d022001-07-12 21:25:22 +00001778#endif
1779 {
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001780 FLAC__ASSERT(best_residual_bits != 0);
1781 break;
Josh Coalson8395d022001-07-12 21:25:22 +00001782 }
1783 sum += 1u << partition_order;
1784 if(best_residual_bits == 0 || residual_bits < best_residual_bits) {
1785 best_residual_bits = residual_bits;
1786 *best_partition_order = partition_order;
1787 best_parameters_index = !best_parameters_index;
1788 }
Josh Coalsonafcd8772001-04-18 22:59:25 +00001789 }
1790 }
Josh Coalson8395d022001-07-12 21:25:22 +00001791 else {
1792 unsigned partition_order;
1793 for(partition_order = min_partition_order; partition_order <= max_partition_order; partition_order++) {
1794#ifdef DONT_ESTIMATE_RICE_BITS
Josh Coalsonf1eff452002-07-31 07:05:33 +00001795 if(!set_partitioned_rice_(abs_residual, residual, residual_samples, predictor_order, rice_parameter, rice_parameter_search_dist, partition_order, private_->parameters[!best_parameters_index], &residual_bits))
Josh Coalson8395d022001-07-12 21:25:22 +00001796#else
Josh Coalsonf1eff452002-07-31 07:05:33 +00001797 if(!set_partitioned_rice_(abs_residual, residual_samples, predictor_order, rice_parameter, rice_parameter_search_dist, partition_order, private_->parameters[!best_parameters_index], &residual_bits))
Josh Coalsonafcd8772001-04-18 22:59:25 +00001798#endif
Josh Coalson8395d022001-07-12 21:25:22 +00001799 {
1800 FLAC__ASSERT(best_residual_bits != 0);
1801 break;
1802 }
1803 if(best_residual_bits == 0 || residual_bits < best_residual_bits) {
1804 best_residual_bits = residual_bits;
1805 *best_partition_order = partition_order;
1806 best_parameters_index = !best_parameters_index;
1807 }
1808 }
1809 }
1810
Josh Coalson8084b052001-11-01 00:27:29 +00001811 memcpy(best_parameters, private_->parameters[best_parameters_index], sizeof(unsigned)*(1<<(*best_partition_order)));
1812 memcpy(best_raw_bits, private_->raw_bits[best_parameters_index], sizeof(unsigned)*(1<<(*best_partition_order)));
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001813
1814 return best_residual_bits;
1815}
1816
Josh Coalsonf1eff452002-07-31 07:05:33 +00001817void precompute_partition_info_sums_(const FLAC__uint32 abs_residual[], FLAC__uint64 abs_residual_partition_sums[], unsigned residual_samples, unsigned predictor_order, unsigned min_partition_order, unsigned max_partition_order)
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001818{
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001819 int partition_order;
Josh Coalsonaef013c2001-04-24 01:25:42 +00001820 unsigned from_partition, to_partition = 0;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001821 const unsigned blocksize = residual_samples + predictor_order;
1822
Josh Coalsonaef013c2001-04-24 01:25:42 +00001823 /* first do max_partition_order */
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001824 for(partition_order = (int)max_partition_order; partition_order >= 0; partition_order--) {
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001825 FLAC__uint64 abs_residual_partition_sum;
Josh Coalson77e3f312001-06-23 03:03:24 +00001826 FLAC__uint32 abs_r;
Josh Coalsonaef013c2001-04-24 01:25:42 +00001827 unsigned partition, partition_sample, partition_samples, residual_sample;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001828 const unsigned partitions = 1u << partition_order;
1829 const unsigned default_partition_samples = blocksize >> partition_order;
1830
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001831 FLAC__ASSERT(default_partition_samples > predictor_order);
1832
1833 for(partition = residual_sample = 0; partition < partitions; partition++) {
1834 partition_samples = default_partition_samples;
1835 if(partition == 0)
1836 partition_samples -= predictor_order;
1837 abs_residual_partition_sum = 0;
1838 for(partition_sample = 0; partition_sample < partition_samples; partition_sample++) {
1839 abs_r = abs_residual[residual_sample];
1840 abs_residual_partition_sum += abs_r;
1841 residual_sample++;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001842 }
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001843 abs_residual_partition_sums[partition] = abs_residual_partition_sum;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001844 }
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001845 to_partition = partitions;
1846 break;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001847 }
Josh Coalsonf76a3612001-04-18 02:28:11 +00001848
Josh Coalson8395d022001-07-12 21:25:22 +00001849 /* now merge partitions for lower orders */
Josh Coalson6bd17572001-05-25 19:02:01 +00001850 for(from_partition = 0, --partition_order; partition_order >= (int)min_partition_order; partition_order--) {
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001851 FLAC__uint64 s;
Josh Coalsonaef013c2001-04-24 01:25:42 +00001852 unsigned i;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001853 const unsigned partitions = 1u << partition_order;
1854 for(i = 0; i < partitions; i++) {
Josh Coalsonaef013c2001-04-24 01:25:42 +00001855 s = abs_residual_partition_sums[from_partition];
Josh Coalsonaef013c2001-04-24 01:25:42 +00001856 from_partition++;
Josh Coalsonaef013c2001-04-24 01:25:42 +00001857 abs_residual_partition_sums[to_partition] = s + abs_residual_partition_sums[from_partition];
Josh Coalsonaef013c2001-04-24 01:25:42 +00001858 from_partition++;
1859 to_partition++;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00001860 }
1861 }
Josh Coalson94e02cd2001-01-25 10:41:06 +00001862}
Josh Coalson8395d022001-07-12 21:25:22 +00001863
Josh Coalsonf1eff452002-07-31 07:05:33 +00001864void precompute_partition_info_escapes_(const FLAC__int32 residual[], unsigned raw_bits_per_partition[], unsigned residual_samples, unsigned predictor_order, unsigned min_partition_order, unsigned max_partition_order)
Josh Coalson8395d022001-07-12 21:25:22 +00001865{
1866 int partition_order;
1867 unsigned from_partition, to_partition = 0;
1868 const unsigned blocksize = residual_samples + predictor_order;
1869
1870 /* first do max_partition_order */
1871 for(partition_order = (int)max_partition_order; partition_order >= 0; partition_order--) {
1872 FLAC__int32 r, residual_partition_min, residual_partition_max;
1873 unsigned silog2_min, silog2_max;
1874 unsigned partition, partition_sample, partition_samples, residual_sample;
1875 const unsigned partitions = 1u << partition_order;
1876 const unsigned default_partition_samples = blocksize >> partition_order;
1877
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001878 FLAC__ASSERT(default_partition_samples > predictor_order);
1879
1880 for(partition = residual_sample = 0; partition < partitions; partition++) {
1881 partition_samples = default_partition_samples;
1882 if(partition == 0)
1883 partition_samples -= predictor_order;
1884 residual_partition_min = residual_partition_max = 0;
1885 for(partition_sample = 0; partition_sample < partition_samples; partition_sample++) {
1886 r = residual[residual_sample];
1887 if(r < residual_partition_min)
1888 residual_partition_min = r;
1889 else if(r > residual_partition_max)
1890 residual_partition_max = r;
1891 residual_sample++;
Josh Coalson8395d022001-07-12 21:25:22 +00001892 }
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001893 silog2_min = FLAC__bitmath_silog2(residual_partition_min);
1894 silog2_max = FLAC__bitmath_silog2(residual_partition_max);
1895 raw_bits_per_partition[partition] = max(silog2_min, silog2_max);
Josh Coalson8395d022001-07-12 21:25:22 +00001896 }
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001897 to_partition = partitions;
1898 break;
Josh Coalson8395d022001-07-12 21:25:22 +00001899 }
1900
1901 /* now merge partitions for lower orders */
1902 for(from_partition = 0, --partition_order; partition_order >= (int)min_partition_order; partition_order--) {
1903 unsigned m;
1904 unsigned i;
1905 const unsigned partitions = 1u << partition_order;
1906 for(i = 0; i < partitions; i++) {
1907 m = raw_bits_per_partition[from_partition];
1908 from_partition++;
1909 raw_bits_per_partition[to_partition] = max(m, raw_bits_per_partition[from_partition]);
1910 from_partition++;
1911 to_partition++;
1912 }
1913 }
1914}
Josh Coalson94e02cd2001-01-25 10:41:06 +00001915
Josh Coalson352e0f62001-03-20 22:55:50 +00001916#ifdef VARIABLE_RICE_BITS
1917#undef VARIABLE_RICE_BITS
1918#endif
Josh Coalson8395d022001-07-12 21:25:22 +00001919#ifndef DONT_ESTIMATE_RICE_BITS
Josh Coalson352e0f62001-03-20 22:55:50 +00001920#define VARIABLE_RICE_BITS(value, parameter) ((value) >> (parameter))
Josh Coalson8395d022001-07-12 21:25:22 +00001921#endif
Josh Coalson352e0f62001-03-20 22:55:50 +00001922
Josh Coalson8395d022001-07-12 21:25:22 +00001923#ifdef DONT_ESTIMATE_RICE_BITS
Josh Coalsonf1eff452002-07-31 07:05:33 +00001924FLAC__bool set_partitioned_rice_(const FLAC__uint32 abs_residual[], const FLAC__int32 residual[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, unsigned parameters[], unsigned *bits)
Josh Coalson8395d022001-07-12 21:25:22 +00001925#else
Josh Coalsonf1eff452002-07-31 07:05:33 +00001926FLAC__bool set_partitioned_rice_(const FLAC__uint32 abs_residual[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, unsigned parameters[], unsigned *bits)
Josh Coalson8395d022001-07-12 21:25:22 +00001927#endif
Josh Coalson94e02cd2001-01-25 10:41:06 +00001928{
Josh Coalson034dfab2001-04-27 19:10:23 +00001929 unsigned rice_parameter, partition_bits;
1930#ifndef NO_RICE_SEARCH
1931 unsigned best_partition_bits;
1932 unsigned min_rice_parameter, max_rice_parameter, best_rice_parameter = 0;
1933#endif
Josh Coalson94e02cd2001-01-25 10:41:06 +00001934 unsigned bits_ = FLAC__ENTROPY_CODING_METHOD_TYPE_LEN + FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ORDER_LEN;
1935
Josh Coalson1b689822001-05-31 20:11:02 +00001936 FLAC__ASSERT(suggested_rice_parameter < FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER);
Josh Coalson2051dd42001-04-12 22:22:34 +00001937
Josh Coalson94e02cd2001-01-25 10:41:06 +00001938 if(partition_order == 0) {
1939 unsigned i;
Josh Coalson352e0f62001-03-20 22:55:50 +00001940
Josh Coalson034dfab2001-04-27 19:10:23 +00001941#ifndef NO_RICE_SEARCH
Josh Coalson60f77d72001-04-25 02:16:36 +00001942 if(rice_parameter_search_dist) {
Josh Coalson034dfab2001-04-27 19:10:23 +00001943 if(suggested_rice_parameter < rice_parameter_search_dist)
Josh Coalson60f77d72001-04-25 02:16:36 +00001944 min_rice_parameter = 0;
1945 else
Josh Coalson034dfab2001-04-27 19:10:23 +00001946 min_rice_parameter = suggested_rice_parameter - rice_parameter_search_dist;
1947 max_rice_parameter = suggested_rice_parameter + rice_parameter_search_dist;
Josh Coalson8395d022001-07-12 21:25:22 +00001948 if(max_rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00001949#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00001950 fprintf(stderr, "clipping rice_parameter (%u -> %u) @2\n", max_rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
1951#endif
Josh Coalson60f77d72001-04-25 02:16:36 +00001952 max_rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
Josh Coalson8395d022001-07-12 21:25:22 +00001953 }
1954 }
1955 else
1956 min_rice_parameter = max_rice_parameter = suggested_rice_parameter;
1957
1958 best_partition_bits = 0xffffffff;
1959 for(rice_parameter = min_rice_parameter; rice_parameter <= max_rice_parameter; rice_parameter++) {
1960#endif
1961#ifdef VARIABLE_RICE_BITS
1962#ifdef FLAC__SYMMETRIC_RICE
1963 partition_bits = (2+rice_parameter) * residual_samples;
1964#else
1965 const unsigned rice_parameter_estimate = rice_parameter-1;
1966 partition_bits = (1+rice_parameter) * residual_samples;
1967#endif
1968#else
1969 partition_bits = 0;
1970#endif
1971 partition_bits += FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_PARAMETER_LEN;
1972 for(i = 0; i < residual_samples; i++) {
1973#ifdef VARIABLE_RICE_BITS
1974#ifdef FLAC__SYMMETRIC_RICE
1975 partition_bits += VARIABLE_RICE_BITS(abs_residual[i], rice_parameter);
1976#else
1977 partition_bits += VARIABLE_RICE_BITS(abs_residual[i], rice_parameter_estimate);
1978#endif
1979#else
1980 partition_bits += FLAC__bitbuffer_rice_bits(residual[i], rice_parameter); /* NOTE: we will need to pass in residual[] in addition to abs_residual[] */
1981#endif
1982 }
1983#ifndef NO_RICE_SEARCH
1984 if(partition_bits < best_partition_bits) {
1985 best_rice_parameter = rice_parameter;
1986 best_partition_bits = partition_bits;
1987 }
1988 }
1989#endif
1990 parameters[0] = best_rice_parameter;
1991 bits_ += best_partition_bits;
1992 }
1993 else {
1994 unsigned partition, residual_sample, save_residual_sample, partition_sample;
Josh Coalsonb3347bd2001-07-16 18:06:41 +00001995 unsigned partition_samples;
1996 FLAC__uint64 mean, k;
Josh Coalson8395d022001-07-12 21:25:22 +00001997 const unsigned partitions = 1u << partition_order;
1998 for(partition = residual_sample = 0; partition < partitions; partition++) {
1999 partition_samples = (residual_samples+predictor_order) >> partition_order;
2000 if(partition == 0) {
2001 if(partition_samples <= predictor_order)
2002 return false;
2003 else
2004 partition_samples -= predictor_order;
2005 }
2006 mean = 0;
2007 save_residual_sample = residual_sample;
2008 for(partition_sample = 0; partition_sample < partition_samples; residual_sample++, partition_sample++)
Josh Coalsonb3347bd2001-07-16 18:06:41 +00002009 mean += abs_residual[residual_sample];
Josh Coalson8395d022001-07-12 21:25:22 +00002010 residual_sample = save_residual_sample;
2011#ifdef FLAC__SYMMETRIC_RICE
2012 mean += partition_samples >> 1; /* for rounding effect */
2013 mean /= partition_samples;
2014
2015 /* calc rice_parameter = floor(log2(mean)) */
2016 rice_parameter = 0;
2017 mean>>=1;
2018 while(mean) {
2019 rice_parameter++;
2020 mean >>= 1;
2021 }
2022#else
2023 /* calc rice_parameter ala LOCO-I */
Josh Coalsonb3347bd2001-07-16 18:06:41 +00002024 for(rice_parameter = 0, k = partition_samples; k < mean; rice_parameter++, k <<= 1)
Josh Coalson8395d022001-07-12 21:25:22 +00002025 ;
2026#endif
2027 if(rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00002028#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00002029 fprintf(stderr, "clipping rice_parameter (%u -> %u) @3\n", rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
2030#endif
2031 rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
2032 }
2033
2034#ifndef NO_RICE_SEARCH
2035 if(rice_parameter_search_dist) {
2036 if(rice_parameter < rice_parameter_search_dist)
2037 min_rice_parameter = 0;
2038 else
2039 min_rice_parameter = rice_parameter - rice_parameter_search_dist;
2040 max_rice_parameter = rice_parameter + rice_parameter_search_dist;
2041 if(max_rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00002042#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00002043 fprintf(stderr, "clipping rice_parameter (%u -> %u) @4\n", max_rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
2044#endif
2045 max_rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
2046 }
2047 }
2048 else
2049 min_rice_parameter = max_rice_parameter = rice_parameter;
2050
2051 best_partition_bits = 0xffffffff;
2052 for(rice_parameter = min_rice_parameter; rice_parameter <= max_rice_parameter; rice_parameter++) {
2053#endif
2054#ifdef VARIABLE_RICE_BITS
2055#ifdef FLAC__SYMMETRIC_RICE
2056 partition_bits = (2+rice_parameter) * partition_samples;
2057#else
2058 const unsigned rice_parameter_estimate = rice_parameter-1;
2059 partition_bits = (1+rice_parameter) * partition_samples;
2060#endif
2061#else
2062 partition_bits = 0;
2063#endif
2064 partition_bits += FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_PARAMETER_LEN;
2065 save_residual_sample = residual_sample;
2066 for(partition_sample = 0; partition_sample < partition_samples; residual_sample++, partition_sample++) {
2067#ifdef VARIABLE_RICE_BITS
2068#ifdef FLAC__SYMMETRIC_RICE
2069 partition_bits += VARIABLE_RICE_BITS(abs_residual[residual_sample], rice_parameter);
2070#else
2071 partition_bits += VARIABLE_RICE_BITS(abs_residual[residual_sample], rice_parameter_estimate);
2072#endif
2073#else
2074 partition_bits += FLAC__bitbuffer_rice_bits(residual[residual_sample], rice_parameter); /* NOTE: we will need to pass in residual[] in addition to abs_residual[] */
2075#endif
2076 }
2077#ifndef NO_RICE_SEARCH
2078 if(rice_parameter != max_rice_parameter)
2079 residual_sample = save_residual_sample;
2080 if(partition_bits < best_partition_bits) {
2081 best_rice_parameter = rice_parameter;
2082 best_partition_bits = partition_bits;
2083 }
2084 }
2085#endif
2086 parameters[partition] = best_rice_parameter;
2087 bits_ += best_partition_bits;
2088 }
2089 }
2090
2091 *bits = bits_;
2092 return true;
2093}
2094
2095#ifdef DONT_ESTIMATE_RICE_BITS
Josh Coalsonf1eff452002-07-31 07:05:33 +00002096FLAC__bool set_partitioned_rice_with_precompute_(const FLAC__int32 residual[], const FLAC__uint64 abs_residual_partition_sums[], const unsigned raw_bits_per_partition[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, const FLAC__bool search_for_escapes, unsigned parameters[], unsigned raw_bits[], unsigned *bits)
Josh Coalson8395d022001-07-12 21:25:22 +00002097#else
Josh Coalsonf1eff452002-07-31 07:05:33 +00002098FLAC__bool set_partitioned_rice_with_precompute_(const FLAC__uint32 abs_residual[], const FLAC__uint64 abs_residual_partition_sums[], const unsigned raw_bits_per_partition[], const unsigned residual_samples, const unsigned predictor_order, const unsigned suggested_rice_parameter, const unsigned rice_parameter_search_dist, const unsigned partition_order, const FLAC__bool search_for_escapes, unsigned parameters[], unsigned raw_bits[], unsigned *bits)
Josh Coalson8395d022001-07-12 21:25:22 +00002099#endif
2100{
2101 unsigned rice_parameter, partition_bits;
2102#ifndef NO_RICE_SEARCH
2103 unsigned best_partition_bits;
2104 unsigned min_rice_parameter, max_rice_parameter, best_rice_parameter = 0;
2105#endif
2106 unsigned flat_bits;
2107 unsigned bits_ = FLAC__ENTROPY_CODING_METHOD_TYPE_LEN + FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ORDER_LEN;
2108
2109 FLAC__ASSERT(suggested_rice_parameter < FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER);
2110
2111 if(partition_order == 0) {
2112 unsigned i;
2113
2114#ifndef NO_RICE_SEARCH
2115 if(rice_parameter_search_dist) {
2116 if(suggested_rice_parameter < rice_parameter_search_dist)
2117 min_rice_parameter = 0;
2118 else
2119 min_rice_parameter = suggested_rice_parameter - rice_parameter_search_dist;
2120 max_rice_parameter = suggested_rice_parameter + rice_parameter_search_dist;
2121 if(max_rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00002122#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00002123 fprintf(stderr, "clipping rice_parameter (%u -> %u) @5\n", max_rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
2124#endif
2125 max_rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
2126 }
Josh Coalson60f77d72001-04-25 02:16:36 +00002127 }
2128 else
Josh Coalson034dfab2001-04-27 19:10:23 +00002129 min_rice_parameter = max_rice_parameter = suggested_rice_parameter;
Josh Coalson2051dd42001-04-12 22:22:34 +00002130
Josh Coalson034dfab2001-04-27 19:10:23 +00002131 best_partition_bits = 0xffffffff;
2132 for(rice_parameter = min_rice_parameter; rice_parameter <= max_rice_parameter; rice_parameter++) {
2133#endif
Josh Coalson352e0f62001-03-20 22:55:50 +00002134#ifdef VARIABLE_RICE_BITS
Josh Coalsonbb6712e2001-04-24 22:54:07 +00002135#ifdef FLAC__SYMMETRIC_RICE
Josh Coalson034dfab2001-04-27 19:10:23 +00002136 partition_bits = (2+rice_parameter) * residual_samples;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002137#else
Josh Coalson352e0f62001-03-20 22:55:50 +00002138 const unsigned rice_parameter_estimate = rice_parameter-1;
Josh Coalson034dfab2001-04-27 19:10:23 +00002139 partition_bits = (1+rice_parameter) * residual_samples;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002140#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00002141#else
2142 partition_bits = 0;
Josh Coalson94e02cd2001-01-25 10:41:06 +00002143#endif
Josh Coalson2051dd42001-04-12 22:22:34 +00002144 partition_bits += FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_PARAMETER_LEN;
Josh Coalson352e0f62001-03-20 22:55:50 +00002145 for(i = 0; i < residual_samples; i++) {
2146#ifdef VARIABLE_RICE_BITS
Josh Coalsonbb6712e2001-04-24 22:54:07 +00002147#ifdef FLAC__SYMMETRIC_RICE
Josh Coalson2051dd42001-04-12 22:22:34 +00002148 partition_bits += VARIABLE_RICE_BITS(abs_residual[i], rice_parameter);
Josh Coalson94e02cd2001-01-25 10:41:06 +00002149#else
Josh Coalson2051dd42001-04-12 22:22:34 +00002150 partition_bits += VARIABLE_RICE_BITS(abs_residual[i], rice_parameter_estimate);
Josh Coalsonb9433f92001-03-17 01:07:00 +00002151#endif
2152#else
Josh Coalson2051dd42001-04-12 22:22:34 +00002153 partition_bits += FLAC__bitbuffer_rice_bits(residual[i], rice_parameter); /* NOTE: we will need to pass in residual[] instead of abs_residual[] */
Josh Coalson94e02cd2001-01-25 10:41:06 +00002154#endif
Josh Coalson2051dd42001-04-12 22:22:34 +00002155 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002156#ifndef NO_RICE_SEARCH
2157 if(partition_bits < best_partition_bits) {
2158 best_rice_parameter = rice_parameter;
2159 best_partition_bits = partition_bits;
Josh Coalson352e0f62001-03-20 22:55:50 +00002160 }
2161 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002162#endif
Josh Coalson8395d022001-07-12 21:25:22 +00002163 if(search_for_escapes) {
2164 flat_bits = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_PARAMETER_LEN + FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_RAW_LEN + raw_bits_per_partition[0] * residual_samples;
2165 if(flat_bits <= best_partition_bits) {
2166 raw_bits[0] = raw_bits_per_partition[0];
2167 best_rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER;
2168 best_partition_bits = flat_bits;
2169 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002170 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002171 parameters[0] = best_rice_parameter;
2172 bits_ += best_partition_bits;
Josh Coalson94e02cd2001-01-25 10:41:06 +00002173 }
2174 else {
Josh Coalson4dacd192001-06-06 21:11:44 +00002175 unsigned partition, residual_sample, save_residual_sample, partition_sample;
Josh Coalsonb3347bd2001-07-16 18:06:41 +00002176 unsigned partition_samples;
2177 FLAC__uint64 mean, k;
Josh Coalsond4e0ddb2001-04-18 02:20:52 +00002178 const unsigned partitions = 1u << partition_order;
Josh Coalson4dacd192001-06-06 21:11:44 +00002179 for(partition = residual_sample = 0; partition < partitions; partition++) {
Josh Coalson94e02cd2001-01-25 10:41:06 +00002180 partition_samples = (residual_samples+predictor_order) >> partition_order;
Josh Coalson034dfab2001-04-27 19:10:23 +00002181 if(partition == 0) {
Josh Coalson94e02cd2001-01-25 10:41:06 +00002182 if(partition_samples <= predictor_order)
2183 return false;
2184 else
2185 partition_samples -= predictor_order;
2186 }
Josh Coalson05d20792001-06-29 23:12:26 +00002187 mean = abs_residual_partition_sums[partition];
Josh Coalsonbb6712e2001-04-24 22:54:07 +00002188#ifdef FLAC__SYMMETRIC_RICE
Josh Coalson05d20792001-06-29 23:12:26 +00002189 mean += partition_samples >> 1; /* for rounding effect */
2190 mean /= partition_samples;
2191
Josh Coalson034dfab2001-04-27 19:10:23 +00002192 /* calc rice_parameter = floor(log2(mean)) */
2193 rice_parameter = 0;
2194 mean>>=1;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002195 while(mean) {
Josh Coalson034dfab2001-04-27 19:10:23 +00002196 rice_parameter++;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002197 mean >>= 1;
2198 }
Josh Coalsonb9433f92001-03-17 01:07:00 +00002199#else
Josh Coalson05d20792001-06-29 23:12:26 +00002200 /* calc rice_parameter ala LOCO-I */
Josh Coalsonb3347bd2001-07-16 18:06:41 +00002201 for(rice_parameter = 0, k = partition_samples; k < mean; rice_parameter++, k <<= 1)
Josh Coalson05d20792001-06-29 23:12:26 +00002202 ;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002203#endif
Josh Coalson8395d022001-07-12 21:25:22 +00002204 if(rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00002205#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00002206 fprintf(stderr, "clipping rice_parameter (%u -> %u) @6\n", rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
2207#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00002208 rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
Josh Coalson8395d022001-07-12 21:25:22 +00002209 }
Josh Coalson60f77d72001-04-25 02:16:36 +00002210
Josh Coalson034dfab2001-04-27 19:10:23 +00002211#ifndef NO_RICE_SEARCH
Josh Coalson60f77d72001-04-25 02:16:36 +00002212 if(rice_parameter_search_dist) {
Josh Coalson034dfab2001-04-27 19:10:23 +00002213 if(rice_parameter < rice_parameter_search_dist)
Josh Coalson60f77d72001-04-25 02:16:36 +00002214 min_rice_parameter = 0;
2215 else
Josh Coalson034dfab2001-04-27 19:10:23 +00002216 min_rice_parameter = rice_parameter - rice_parameter_search_dist;
2217 max_rice_parameter = rice_parameter + rice_parameter_search_dist;
Josh Coalson8395d022001-07-12 21:25:22 +00002218 if(max_rice_parameter >= FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER) {
Josh Coalson31209492001-07-18 23:43:01 +00002219#ifdef DEBUG_VERBOSE
Josh Coalson8395d022001-07-12 21:25:22 +00002220 fprintf(stderr, "clipping rice_parameter (%u -> %u) @7\n", max_rice_parameter, FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1);
2221#endif
Josh Coalson60f77d72001-04-25 02:16:36 +00002222 max_rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER - 1;
Josh Coalson8395d022001-07-12 21:25:22 +00002223 }
Josh Coalson60f77d72001-04-25 02:16:36 +00002224 }
2225 else
2226 min_rice_parameter = max_rice_parameter = rice_parameter;
Josh Coalson60f77d72001-04-25 02:16:36 +00002227
Josh Coalson034dfab2001-04-27 19:10:23 +00002228 best_partition_bits = 0xffffffff;
2229 for(rice_parameter = min_rice_parameter; rice_parameter <= max_rice_parameter; rice_parameter++) {
2230#endif
Josh Coalson352e0f62001-03-20 22:55:50 +00002231#ifdef VARIABLE_RICE_BITS
Josh Coalsonbb6712e2001-04-24 22:54:07 +00002232#ifdef FLAC__SYMMETRIC_RICE
Josh Coalson034dfab2001-04-27 19:10:23 +00002233 partition_bits = (2+rice_parameter) * partition_samples;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002234#else
Josh Coalson034dfab2001-04-27 19:10:23 +00002235 const unsigned rice_parameter_estimate = rice_parameter-1;
2236 partition_bits = (1+rice_parameter) * partition_samples;
Josh Coalsonb9433f92001-03-17 01:07:00 +00002237#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00002238#else
2239 partition_bits = 0;
Josh Coalson94e02cd2001-01-25 10:41:06 +00002240#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00002241 partition_bits += FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_PARAMETER_LEN;
Josh Coalson4dacd192001-06-06 21:11:44 +00002242 save_residual_sample = residual_sample;
2243 for(partition_sample = 0; partition_sample < partition_samples; residual_sample++, partition_sample++) {
Josh Coalson352e0f62001-03-20 22:55:50 +00002244#ifdef VARIABLE_RICE_BITS
Josh Coalsonbb6712e2001-04-24 22:54:07 +00002245#ifdef FLAC__SYMMETRIC_RICE
Josh Coalson4dacd192001-06-06 21:11:44 +00002246 partition_bits += VARIABLE_RICE_BITS(abs_residual[residual_sample], rice_parameter);
Josh Coalson94e02cd2001-01-25 10:41:06 +00002247#else
Josh Coalson4dacd192001-06-06 21:11:44 +00002248 partition_bits += VARIABLE_RICE_BITS(abs_residual[residual_sample], rice_parameter_estimate);
Josh Coalsonb9433f92001-03-17 01:07:00 +00002249#endif
2250#else
Josh Coalson4dacd192001-06-06 21:11:44 +00002251 partition_bits += FLAC__bitbuffer_rice_bits(residual[residual_sample], rice_parameter); /* NOTE: we will need to pass in residual[] instead of abs_residual[] */
Josh Coalson94e02cd2001-01-25 10:41:06 +00002252#endif
Josh Coalson034dfab2001-04-27 19:10:23 +00002253 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002254#ifndef NO_RICE_SEARCH
Josh Coalson4dacd192001-06-06 21:11:44 +00002255 if(rice_parameter != max_rice_parameter)
2256 residual_sample = save_residual_sample;
Josh Coalson034dfab2001-04-27 19:10:23 +00002257 if(partition_bits < best_partition_bits) {
2258 best_rice_parameter = rice_parameter;
2259 best_partition_bits = partition_bits;
2260 }
Josh Coalson2051dd42001-04-12 22:22:34 +00002261 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002262#endif
Josh Coalson8395d022001-07-12 21:25:22 +00002263 if(search_for_escapes) {
2264 flat_bits = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_PARAMETER_LEN + FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_RAW_LEN + raw_bits_per_partition[partition] * partition_samples;
2265 if(flat_bits <= best_partition_bits) {
2266 raw_bits[partition] = raw_bits_per_partition[partition];
2267 best_rice_parameter = FLAC__ENTROPY_CODING_METHOD_PARTITIONED_RICE_ESCAPE_PARAMETER;
2268 best_partition_bits = flat_bits;
2269 }
Josh Coalson2051dd42001-04-12 22:22:34 +00002270 }
Josh Coalson034dfab2001-04-27 19:10:23 +00002271 parameters[partition] = best_rice_parameter;
2272 bits_ += best_partition_bits;
Josh Coalson94e02cd2001-01-25 10:41:06 +00002273 }
2274 }
2275
2276 *bits = bits_;
2277 return true;
2278}
Josh Coalson859bc542001-03-27 22:22:27 +00002279
Josh Coalsonf1eff452002-07-31 07:05:33 +00002280unsigned get_wasted_bits_(FLAC__int32 signal[], unsigned samples)
Josh Coalson859bc542001-03-27 22:22:27 +00002281{
2282 unsigned i, shift;
Josh Coalson77e3f312001-06-23 03:03:24 +00002283 FLAC__int32 x = 0;
Josh Coalson859bc542001-03-27 22:22:27 +00002284
2285 for(i = 0; i < samples && !(x&1); i++)
2286 x |= signal[i];
2287
2288 if(x == 0) {
2289 shift = 0;
2290 }
2291 else {
2292 for(shift = 0; !(x&1); shift++)
2293 x >>= 1;
2294 }
2295
2296 if(shift > 0) {
2297 for(i = 0; i < samples; i++)
2298 signal[i] >>= shift;
2299 }
2300
2301 return shift;
2302}
Josh Coalsond86e03b2002-08-03 21:56:15 +00002303
2304void append_to_verify_fifo_(verify_input_fifo *fifo, const FLAC__int32 * const input[], unsigned input_offset, unsigned channels, unsigned wide_samples)
2305{
2306 unsigned channel;
2307
2308 for(channel = 0; channel < channels; channel++)
2309 memcpy(&fifo->data[channel][fifo->tail], &input[channel][input_offset], sizeof(FLAC__int32) * wide_samples);
2310
2311 fifo->tail += wide_samples;
2312
2313 FLAC__ASSERT(fifo->tail <= fifo->size);
2314}
2315
2316void append_to_verify_fifo_interleaved_(verify_input_fifo *fifo, const FLAC__int32 input[], unsigned input_offset, unsigned channels, unsigned wide_samples)
2317{
2318 unsigned channel;
2319 unsigned sample, wide_sample;
2320 unsigned tail = fifo->tail;
2321
2322 sample = input_offset * channels;
2323 for(wide_sample = 0; wide_sample < wide_samples; wide_sample++) {
2324 for(channel = 0; channel < channels; channel++)
2325 fifo->data[channel][tail] = input[sample++];
2326 tail++;
2327 }
2328 fifo->tail = tail;
2329
2330 FLAC__ASSERT(fifo->tail <= fifo->size);
2331}
2332
2333FLAC__StreamDecoderReadStatus verify_read_callback_(const FLAC__StreamDecoder *decoder, FLAC__byte buffer[], unsigned *bytes, void *client_data)
2334{
2335 FLAC__StreamEncoder *encoder = (FLAC__StreamEncoder*)client_data;
2336 const unsigned encoded_bytes = encoder->private_->verify.output.bytes;
2337 (void)decoder;
2338
2339 if(encoder->private_->verify.needs_magic_hack) {
2340 FLAC__ASSERT(*bytes >= FLAC__STREAM_SYNC_LENGTH);
2341 *bytes = FLAC__STREAM_SYNC_LENGTH;
2342 memcpy(buffer, FLAC__STREAM_SYNC_STRING, *bytes);
2343 encoder->private_->verify.needs_magic_hack = false;
2344 }
2345 else {
2346 if(encoded_bytes == 0) {
2347 //@@@@ underflow happened, should we do something else here? is this an assert failure?
2348 return FLAC__STREAM_DECODER_READ_STATUS_ABORT;
2349 }
2350 else if(encoded_bytes < *bytes)
2351 *bytes = encoded_bytes;
2352 memcpy(buffer, encoder->private_->verify.output.data, *bytes);
2353 encoder->private_->verify.output.data += *bytes;
2354 encoder->private_->verify.output.bytes -= *bytes;
2355 }
2356
2357 return FLAC__STREAM_DECODER_READ_STATUS_CONTINUE;
2358}
2359
2360FLAC__StreamDecoderWriteStatus verify_write_callback_(const FLAC__StreamDecoder *decoder, const FLAC__Frame *frame, const FLAC__int32 * const buffer[], void *client_data)
2361{
2362 FLAC__StreamEncoder *encoder = (FLAC__StreamEncoder *)client_data;
2363 unsigned channel;
2364 const unsigned channels = FLAC__stream_decoder_get_channels(decoder);
2365 const unsigned blocksize = frame->header.blocksize;
2366 const unsigned bytes_per_block = sizeof(FLAC__int32) * blocksize;
2367
2368 for(channel = 0; channel < channels; channel++) {
2369 if(0 != memcmp(buffer[channel], encoder->private_->verify.input_fifo.data[channel], bytes_per_block)) {
2370 unsigned i, sample = 0;
2371 FLAC__int32 expect = 0, got = 0;
2372
2373 for(i = 0; i < blocksize; i++) {
2374 if(buffer[channel][i] != encoder->private_->verify.input_fifo.data[channel][i]) {
2375 sample = i;
2376 expect = (FLAC__int32)encoder->private_->verify.input_fifo.data[channel][i];
2377 got = (FLAC__int32)buffer[channel][i];
2378 break;
2379 }
2380 }
2381 FLAC__ASSERT(i < blocksize);
2382 FLAC__ASSERT(frame->header.number_type == FLAC__FRAME_NUMBER_TYPE_SAMPLE_NUMBER);
2383 encoder->private_->verify.error_stats.absolute_sample = frame->header.number.sample_number + sample;
2384 encoder->private_->verify.error_stats.frame_number = frame->header.number.sample_number / blocksize;
2385 encoder->private_->verify.error_stats.channel = channel;
2386 encoder->private_->verify.error_stats.sample = sample;
2387 encoder->private_->verify.error_stats.expected = expect;
2388 encoder->private_->verify.error_stats.got = got;
2389 encoder->protected_->state = FLAC__STREAM_ENCODER_VERIFY_MISMATCH_IN_AUDIO_DATA;
2390 return FLAC__STREAM_DECODER_WRITE_STATUS_ABORT;
2391 }
2392 }
2393 /* dequeue the frame from the fifo */
2394 for(channel = 0; channel < channels; channel++) {
2395 memmove(&encoder->private_->verify.input_fifo.data[channel][0], &encoder->private_->verify.input_fifo.data[channel][blocksize], encoder->private_->verify.input_fifo.tail - blocksize);
2396 }
2397 encoder->private_->verify.input_fifo.tail -= blocksize;
2398 return FLAC__STREAM_DECODER_WRITE_STATUS_CONTINUE;
2399}
2400
2401void verify_metadata_callback_(const FLAC__StreamDecoder *decoder, const FLAC__StreamMetadata *metadata, void *client_data)
2402{
2403 (void)decoder, (void)metadata, (void)client_data;
2404}
2405
2406void verify_error_callback_(const FLAC__StreamDecoder *decoder, FLAC__StreamDecoderErrorStatus status, void *client_data)
2407{
2408 FLAC__StreamEncoder *encoder = (FLAC__StreamEncoder*)client_data;
2409 (void)decoder, (void)status;
2410 encoder->protected_->state = FLAC__STREAM_ENCODER_VERIFY_DECODER_ERROR;
2411}