|  | /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ | 
|  | /* This Source Code Form is subject to the terms of the Mozilla Public | 
|  | * License, v. 2.0. If a copy of the MPL was not distributed with this | 
|  | * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ | 
|  |  | 
|  | /* | 
|  | ** File:        append.c | 
|  | ** Description: Testing File writes where PR_APPEND was used on open | 
|  | ** | 
|  | ** append attempts to verify that a file opened with PR_APPEND | 
|  | ** will always append to the end of file, regardless where the | 
|  | ** current file pointer is positioned. To do this, PR_Seek() is | 
|  | ** called before each write with the position set to beginning of | 
|  | ** file. Subsequent writes should always append. | 
|  | ** The file is read back, summing the integer data written to the | 
|  | ** file. If the expected result is equal, the test passes. | 
|  | ** | 
|  | ** See BugSplat: 4090 | 
|  | */ | 
|  | #include "plgetopt.h" | 
|  | #include "nspr.h" | 
|  |  | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  |  | 
|  | PRIntn  debug = 0; | 
|  | PRIntn  verbose = 0; | 
|  | PRBool  failedAlready = PR_FALSE; | 
|  | const PRInt32 addedBytes = 1000; | 
|  | const PRInt32   buf = 1; /* constant written to fd, addedBytes times */ | 
|  | PRInt32         inBuf;   /* read it back into here */ | 
|  |  | 
|  | int main(int argc, char **argv) | 
|  | { | 
|  | PRStatus    rc; | 
|  | PRInt32     rv; | 
|  | PRFileDesc  *fd; | 
|  | PRIntn      i; | 
|  | PRInt32     sum = 0; | 
|  |  | 
|  | {   /* Get command line options */ | 
|  | PLOptStatus os; | 
|  | PLOptState *opt = PL_CreateOptState(argc, argv, "vd"); | 
|  |  | 
|  | while (PL_OPT_EOL != (os = PL_GetNextOpt(opt))) | 
|  | { | 
|  | if (PL_OPT_BAD == os) { | 
|  | continue; | 
|  | } | 
|  | switch (opt->option) | 
|  | { | 
|  | case 'd':  /* debug */ | 
|  | debug = 1; | 
|  | break; | 
|  | case 'v':  /* verbose */ | 
|  | verbose = 1; | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  | } | 
|  | PL_DestroyOptState(opt); | 
|  | } /* end block "Get command line options" */ | 
|  | /* ---------------------------------------------------------------------- */ | 
|  | fd = PR_Open( "./tmp-nsprAppend", (PR_APPEND | PR_CREATE_FILE | PR_TRUNCATE | PR_WRONLY), 0666 ); | 
|  | if ( NULL == fd )  { | 
|  | if (debug) { | 
|  | printf("PR_Open() failed for writing: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  |  | 
|  | for ( i = 0; i < addedBytes ; i++ ) { | 
|  | rv = PR_Write( fd, &buf, sizeof(buf)); | 
|  | if ( sizeof(buf) != rv )  { | 
|  | if (debug) { | 
|  | printf("PR_Write() failed: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  | rv = PR_Seek( fd, 0, PR_SEEK_SET ); | 
|  | if ( -1 == rv )  { | 
|  | if (debug) { | 
|  | printf("PR_Seek() failed: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  | } | 
|  | rc = PR_Close( fd ); | 
|  | if ( PR_FAILURE == rc ) { | 
|  | if (debug) { | 
|  | printf("PR_Close() failed after writing: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  | /* ---------------------------------------------------------------------- */ | 
|  | fd = PR_Open( "./tmp-nsprAppend", PR_RDONLY, 0 ); | 
|  | if ( NULL == fd )  { | 
|  | if (debug) { | 
|  | printf("PR_Open() failed for reading: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  |  | 
|  | for ( i = 0; i < addedBytes ; i++ ) { | 
|  | rv = PR_Read( fd, &inBuf, sizeof(inBuf)); | 
|  | if ( sizeof(inBuf) != rv)  { | 
|  | if (debug) { | 
|  | printf("PR_Write() failed: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  | sum += inBuf; | 
|  | } | 
|  |  | 
|  | rc = PR_Close( fd ); | 
|  | if ( PR_FAILURE == rc ) { | 
|  | if (debug) { | 
|  | printf("PR_Close() failed after reading: %d\n", PR_GetError()); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  | if ( sum != addedBytes )  { | 
|  | if (debug) { | 
|  | printf("Uh Oh! addedBytes: %d. Sum: %d\n", addedBytes, sum); | 
|  | } | 
|  | failedAlready = PR_TRUE; | 
|  | goto Finished; | 
|  | } | 
|  |  | 
|  | /* ---------------------------------------------------------------------- */ | 
|  | Finished: | 
|  | if (debug || verbose) { | 
|  | printf("%s\n", (failedAlready)? "FAILED" : "PASSED" ); | 
|  | } | 
|  | return( (failedAlready)? 1 : 0 ); | 
|  | }  /* main() */ | 
|  |  | 
|  | /* append.c */ |