1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License, Version 1.0 only 6 * (the "License"). You may not use this file except in compliance 7 * with the License. 8 * 9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10 * or http://www.opensolaris.org/os/licensing. 11 * See the License for the specific language governing permissions 12 * and limitations under the License. 13 * 14 * When distributing Covered Code, include this CDDL HEADER in each 15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16 * If applicable, add the following below this CDDL HEADER, with the 17 * fields enclosed by brackets "[]" replaced with your own identifying 18 * information: Portions Copyright [yyyy] [name of copyright owner] 19 * 20 * CDDL HEADER END 21 */ 22 /* 23 * Copyright (c) 1992-2001 by Sun Microsystems, Inc. 24 * All rights reserved. 25 */ 26 27 #include <stdlib.h> 28 #include <memory.h> 29 #include "../include/AudioDebug.h" 30 #include "../include/AudioBuffer.h" 31 #include "../include/zmalloc.h" 32 33 // class AudioBuffer methods 34 35 // Constructor with optional hdr, size, and name arguments 36 AudioBuffer:: 37 AudioBuffer( 38 double len, // buffer length, in seconds 39 const char *local_name): // name 40 AudioStream(local_name), buflen(len), zflag(0), bufsize(0), bufaddr(0) 41 { 42 } 43 44 // Destructor 45 AudioBuffer:: 46 ~AudioBuffer() 47 { 48 (void) SetSize(0.); // deallocate the buffer 49 } 50 51 // XXX - the following functions are good candidates for inlining 52 53 // Return TRUE if the stream is 'open' 54 Boolean AudioBuffer:: 55 opened() const 56 { 57 // A buffer is open if it is allocated and has a valid header 58 return (hdrset() && (GetAddress() != 0)); 59 } 60 61 #define MIN_ZBUFFER (8192 * 10) // only for large buffers 62 63 // Allocate buffer. Size and header must be set. 64 AudioError AudioBuffer:: 65 alloc() 66 { 67 long size; 68 size_t cnt; 69 unsigned int ncpy; 70 void* tmpbuf; 71 72 // this is going to be the size we're setting the buffer 73 // to (buflen field). it's set by calling SetSize(). 74 size = GetHeader().Time_to_Bytes(GetSize()); 75 76 // this is actual current size, in bytes, of the allocated 77 // buffer (the bufsize field). 78 cnt = GetByteCount(); 79 80 AUDIO_DEBUG((5, "%d: AudioBuffer::alloc - change from %d to %d bytes\n", 81 getid(), cnt, size)); 82 83 bufsize = 0; 84 85 if (size == 0) { 86 // Zero size deletes the buffer 87 if (bufaddr != 0) { 88 if (zflag != 0) { 89 AUDIO_DEBUG((5, 90 "%d: AudioBuffer::alloc - zfree mmapped buffer\n", 91 getid())); 92 (void) zfree((char *)bufaddr); 93 } else { 94 AUDIO_DEBUG((5, 95 "%d: AudioBuffer::alloc - free malloc'd buffer\n", 96 getid())); 97 (void) free((char *)bufaddr); 98 } 99 zflag = 0; 100 } 101 bufaddr = 0; 102 103 } else if (size < 0) { 104 // Ridiculous size 105 AUDIO_DEBUG((5, "%d: AudioBuffer::alloc - bad size\n", 106 getid())); 107 return (RaiseError(AUDIO_ERR_BADARG)); 108 109 } else if (bufaddr == 0) { 110 // Allocate a new buffer 111 if (size > MIN_ZBUFFER) { 112 AUDIO_DEBUG((5, 113 "%d: AudioBuffer::alloc - zmalloc new buffer\n", 114 getid())); 115 bufaddr = (void*) zmalloc((unsigned int)size); 116 zflag = 1; 117 } else { 118 AUDIO_DEBUG((5, 119 "%d: AudioBuffer::alloc - malloc new buffer\n", 120 getid())); 121 bufaddr = (void*) malloc((unsigned int)size); 122 zflag = 0; 123 } 124 if (bufaddr == 0) { 125 AUDIO_DEBUG((5, 126 "%d: AudioBuffer::alloc - buffer alloc failed\n", 127 getid())); 128 return (RaiseError(AUDIO_UNIXERROR)); 129 } 130 } else { 131 // A buffer was already allocated. 132 // Change its size, preserving as much data as possible. 133 if ((cnt <= MIN_ZBUFFER) && (size <= MIN_ZBUFFER) && 134 (zflag == 0)) { 135 AUDIO_DEBUG((5, 136 "%d: AudioBuffer::alloc - realloc to change size\n", 137 getid())); 138 bufaddr = (void*) 139 realloc((char *)bufaddr, (unsigned int)size); 140 } else { 141 AUDIO_DEBUG((5, 142 "%d: AudioBuffer::alloc - zmalloc new buffer\n", 143 getid())); 144 tmpbuf = bufaddr; 145 bufaddr = (void*) zmalloc((unsigned int)size); 146 147 // copy over as much of the old data as will fit 148 if (bufaddr != 0) { 149 ncpy = (cnt < size) ? (unsigned int)cnt : 150 (unsigned int)size; 151 152 AUDIO_DEBUG((5, 153 "%d: AudioBuffer::alloc - trasnfer %d bytes\n", 154 getid(), ncpy)); 155 (void) memcpy(bufaddr, tmpbuf, ncpy); 156 } 157 if ((cnt > MIN_ZBUFFER) && (zflag != 0)) { 158 AUDIO_DEBUG((5, 159 "%d: AudioBuffer::alloc - zfree old buffer\n", 160 getid())); 161 (void) zfree((char *)tmpbuf); 162 } else { 163 AUDIO_DEBUG((5, 164 "%d: AudioBuffer::alloc - free old buffer\n", 165 getid())); 166 (void) free((char *)tmpbuf); 167 } 168 zflag = 1; 169 } 170 if (bufaddr == 0) { 171 return (RaiseError(AUDIO_UNIXERROR)); 172 } 173 } 174 bufsize = (size_t)size; 175 return (AUDIO_SUCCESS); 176 } 177 178 179 // Return the buffer address 180 void* AudioBuffer:: 181 GetAddress() const 182 { 183 return (GetAddress(0.)); 184 } 185 186 // Return the buffer address at a given time offset 187 // Returns NULL if no buffer, or the position is not within the buffer. 188 void* AudioBuffer:: 189 GetAddress( 190 Double pos) const 191 { 192 char *addr; 193 AudioHdr hdr_local; 194 AudioHdr(AudioBuffer::*hfunc)()const; 195 196 addr = (char *)bufaddr; 197 if ((addr == 0) || (pos < 0.) || (pos >= buflen)) 198 return (NULL); 199 200 // If no offset, it's ok if the header hasn't been set yet 201 if (pos == 0.) 202 return ((void*) addr); 203 204 // Get the header and make sure it's valid 205 // This convoluted hfunc works around non-const function problems 206 hfunc = (AudioHdr(AudioBuffer::*)() const)&AudioBuffer::GetHeader; 207 hdr_local = (this->*hfunc)(); 208 if (hdr_local.Validate()) 209 return (NULL); 210 addr += hdr_local.Time_to_Bytes(pos); 211 212 // One more validation, to be paranoid before handing out this address 213 if (addr >= ((char *)bufaddr + bufsize)) 214 return (NULL); 215 return ((void*) addr); 216 } 217 218 // Return the buffer size, in bytes 219 // (as opposed to 'length' which indicates how much data is in the buffer) 220 size_t AudioBuffer:: 221 GetByteCount() const 222 { 223 return (bufsize); 224 } 225 226 // Return the buffer size, in seconds 227 // (as opposed to 'length' which indicates how much data is in the buffer) 228 Double AudioBuffer:: 229 GetSize() const 230 { 231 return (buflen); 232 } 233 234 // Set the buffer size, allocating the buffer as necessary 235 AudioError AudioBuffer:: 236 SetSize( 237 Double len) // new size, in seconds 238 { 239 // If no change in size, do nothing 240 if (len == buflen) 241 return (AUDIO_SUCCESS); 242 243 // If header not set, store the size for later 244 buflen = len; 245 if (!hdrset()) { 246 return (AUDIO_SUCCESS); 247 } 248 249 // If shrinking buffer, note this 250 if (buflen < GetLength()) 251 SetLength(buflen); 252 return (alloc()); 253 } 254 255 // Set the data header 256 // If no buffer allocated, allocate one now (if size is set). 257 // If buffer allocated, fiddle the sizes to account for new header type. 258 AudioError AudioBuffer:: 259 SetHeader( 260 const AudioHdr& h) // header to copy 261 { 262 AudioError err; 263 264 // Validate, then update the header 265 err = h.Validate(); 266 if (err) 267 return (RaiseError(err)); 268 (void) AudioStream::updateheader(h); 269 270 // If no size set, done for now 271 if (buflen == 0.) 272 return (AUDIO_SUCCESS); 273 274 // If no buffer allocated, allocate one now 275 if (GetAddress() == 0) 276 return (alloc()); 277 278 // If buffer allocated, change size to match new header 279 buflen = h.Bytes_to_Time(GetByteCount()); 280 return (AUDIO_SUCCESS); 281 } 282 283 // Set the buffer length (ie, the amount of data written to the buffer) 284 void AudioBuffer:: 285 SetLength( 286 Double len) // new length 287 { 288 if (!hdrset() || (len < 0.)) // no-op if not ready 289 return; 290 if (!opened() && (len > 0.)) 291 return; 292 293 if (Undefined(len) || (len > GetSize())) { 294 // Limit to the size of the buffer 295 setlength(GetSize()); 296 } else { 297 setlength(len); 298 } 299 } 300 301 // Copy data from local buffer into specified buffer. 302 // No data format translation takes place. 303 // The object's read position is not updated. 304 AudioError AudioBuffer:: 305 ReadData( 306 void* buf, // destination buffer address 307 size_t& len, // buffer length (updated) 308 Double& pos) // start position (updated) 309 { 310 off_t resid; 311 off_t cnt; 312 off_t offset; 313 AudioError err; 314 315 // Copy length, zero return value 316 cnt = (off_t)len; 317 len = 0; 318 319 // Cannot read if buffer or header not valid 320 if (!opened()) 321 return (RaiseError(AUDIO_ERR_NOEFFECT)); 322 323 // Position must be valid 324 if ((pos < 0.) || (cnt < 0)) 325 return (RaiseError(AUDIO_ERR_BADARG)); 326 327 // If the starting offset is at or beyond EOF, return eof flag 328 if (pos >= GetLength()) { 329 err = AUDIO_EOF; 330 err.sys = AUDIO_COPY_INPUT_EOF; 331 return (err); 332 } 333 334 // Limit transfer to remaining room in buffer 335 offset = GetHeader().Time_to_Bytes(pos); 336 resid = GetHeader().Time_to_Bytes(GetLength()) - offset; 337 if (resid <= 0) { 338 err = AUDIO_EOF; 339 err.sys = AUDIO_COPY_INPUT_EOF; 340 return (err); 341 } 342 if (cnt > resid) 343 cnt = resid; 344 345 // Fix the alignment to make sure we're not splitting frames 346 err = AUDIO_SUCCESS; 347 if (GetHeader().Bytes_to_Bytes(cnt) > 0) { 348 // Copy as much data as possible 349 memcpy((char *)buf, (char *)((off_t)GetAddress() + offset), 350 (int)cnt); 351 } else { 352 err.sys = AUDIO_COPY_ZERO_LIMIT; 353 } 354 355 // Return the updated transfer size and position 356 len = (size_t)cnt; 357 pos = GetHeader().Bytes_to_Time(offset + cnt); 358 359 360 // Check to see if the endian is right. 361 coerceEndian((unsigned char *)buf, len, localByteOrder()); 362 363 return (err); 364 } 365 366 // Copy data to local buffer from specified buffer. 367 // No data format translation takes place. 368 // The object's write position is not updated. 369 AudioError AudioBuffer:: 370 WriteData( 371 void* buf, // source buffer address 372 size_t& len, // buffer length (updated) 373 Double& pos) // start position (updated) 374 { 375 off_t resid; 376 off_t cnt; 377 off_t offset; 378 AudioError err; 379 380 // Copy length, zero return value 381 cnt = (off_t)len; 382 len = 0; 383 384 // Cannot write if buffer or header not valid 385 if (!opened()) 386 return (RaiseError(AUDIO_ERR_NOEFFECT)); 387 388 // Position must be valid 389 if ((pos < 0.) || (cnt < 0)) 390 return (RaiseError(AUDIO_ERR_BADARG)); 391 392 // If the starting offset beyond end of buffer, return short write flag 393 if (pos >= GetSize()) { 394 err = AUDIO_EOF; 395 err.sys = AUDIO_COPY_OUTPUT_EOF; 396 return (err); 397 } 398 399 // Limit transfer to remaining room in buffer 400 offset = GetHeader().Time_to_Bytes(pos); 401 resid = (off_t)bufsize - offset; 402 if (resid <= 0) { 403 err = AUDIO_EOF; 404 err.sys = AUDIO_COPY_OUTPUT_EOF; 405 return (err); 406 } 407 if (cnt > resid) 408 cnt = resid; 409 410 // Fix the alignment to make sure we're not splitting frames 411 err = AUDIO_SUCCESS; 412 if (GetHeader().Bytes_to_Bytes(cnt) > 0) { 413 // Copy as much data as possible 414 memcpy((char *)((off_t)GetAddress() + offset), (char *)buf, 415 (int)cnt); 416 } else { 417 err.sys = AUDIO_COPY_ZERO_LIMIT; 418 } 419 420 // Return the updated transfer size and position 421 len = (size_t)cnt; 422 pos = GetHeader().Bytes_to_Time(offset + cnt); 423 424 // The end of a write to a buffer always becomes the buffer EOF 425 setlength(pos); 426 return (err); 427 } 428 429 // AppendData is just like WriteData, except that it guarantees to extend 430 // the buffer if it is not big enough. 431 // The object's write position is not updated. 432 AudioError AudioBuffer:: 433 AppendData( 434 void* buf, // source buffer address 435 size_t& len, // buffer length (updated) 436 Double& pos) // start position (updated) 437 { 438 Double local_length; 439 AudioError err; 440 441 // Cannot write if header not valid 442 if (!hdrset()) 443 return (RaiseError(AUDIO_ERR_NOEFFECT)); 444 445 // Position must be valid 446 if (pos < 0.) 447 return (RaiseError(AUDIO_ERR_BADARG)); 448 449 // If the ending offset is beyond end of buffer, extend it 450 local_length = pos + GetHeader().Bytes_to_Time(len); 451 if (local_length > GetSize()) { 452 err = SetSize(local_length); 453 if (err != AUDIO_SUCCESS) 454 return (err); 455 } 456 return (WriteData(buf, len, pos)); 457 } 458 459 // Copy routine to copy direct to destination 460 AudioError AudioBuffer:: 461 AsyncCopy( 462 Audio* to, // audio object to copy to 463 Double& frompos, 464 Double& topos, 465 Double& limit) 466 { 467 caddr_t bptr; 468 size_t cnt; 469 size_t svcnt; 470 Double svfrom; 471 Double svto; 472 Double lim; 473 AudioHdr tohdr; 474 AudioError err; 475 476 // Cannot write if buffer or header not valid 477 if (!opened()) 478 return (RaiseError(AUDIO_ERR_NOEFFECT)); 479 480 tohdr = to->GetHeader(); 481 if (limit < 0.) 482 return (RaiseError(AUDIO_ERR_BADARG)); 483 484 // Get maximum possible copy length 485 svfrom = GetLength(); 486 if (frompos >= svfrom) { 487 limit = 0.; 488 err = AUDIO_EOF; 489 err.sys = AUDIO_COPY_INPUT_EOF; 490 return (err); 491 } 492 lim = svfrom - frompos; 493 if (!Undefined(limit) && (limit < lim)) 494 lim = limit; 495 496 limit = 0.; 497 498 bptr = (caddr_t)GetAddress(frompos); 499 if (bptr == 0) { 500 err = AUDIO_EOF; 501 err.sys = AUDIO_COPY_INPUT_EOF; 502 return (err); 503 } 504 cnt = (size_t)GetHeader().Time_to_Bytes(lim); 505 if (cnt == 0) { 506 err = AUDIO_SUCCESS; 507 err.sys = AUDIO_COPY_ZERO_LIMIT; 508 return (err); 509 } 510 511 // Add a bunch of paranoid checks 512 svcnt = (size_t)GetAddress() + (size_t)GetByteCount(); 513 if ((bptr + cnt) > (caddr_t)svcnt) { 514 // re-adjust cnt so it reads up to the end of file 515 cnt = (size_t)((caddr_t)svcnt - bptr); 516 } 517 if (GetHeader().Bytes_to_Bytes(cnt) == 0) { 518 err = AUDIO_EOF; 519 err.sys = AUDIO_COPY_INPUT_EOF; 520 return (err); 521 } 522 523 // Write the data to the destination and update pointers/ctrs 524 svfrom = frompos; 525 svto = topos; 526 svcnt = cnt; 527 err = to->WriteData(bptr, cnt, topos); 528 limit = topos - svto; 529 frompos = svfrom + limit; 530 531 // Report short writes 532 if (!err && (cnt < svcnt)) { 533 err.sys = AUDIO_COPY_SHORT_OUTPUT; 534 } 535 return (err); 536 } 537