Implemented large parts of the stdio(3), including fprintf.
Made FILE an interface to various backends. This allows application writers
to override the standard FILE API functions with their own backends. This
is highly unportable - it'd be nice if a real standard existed for this.
glibc already does something like this internally, but AFAIK you can't hook
into it.
Added fdopen(3), fopen(3), fregister(3), funregister(3), fread(3),
fwrite(3), fseek(3), clearerr(3), ferror(3), feof(3), rewind(3), ftell(3),
fflush(3), fclose(3), fileno(3), fnewline(3), fcloseall(3), memset(3),
stdio(3), vfprintf(3), fprintf(3), and vprintf(3).
Added a file-descriptor backend to the FILE API.
fd's {0, 1, 2} are now initialized as stdin, stdout, and stderr when the
standard library initializes.
fcloseall(3) is now called on exit(3).
decl/intn_t_.h now @include(size_t.h) instead of declaring it itself.
Added <stdint.h>.
The following programs now flush stdout: cat(1), clear(1), editor(1),
init(1), mxsh(1).
printf(3) is now hooked up against vprintf(3), while Maxsi::PrintF
remains using the system call, for now.
2011-12-23 22:08:10 -05:00
|
|
|
/******************************************************************************
|
|
|
|
|
|
|
|
COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011.
|
|
|
|
|
|
|
|
This file is part of LibMaxsi.
|
|
|
|
|
|
|
|
LibMaxsi is free software: you can redistribute it and/or modify it under
|
|
|
|
the terms of the GNU Lesser General Public License as published by the Free
|
|
|
|
Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
LibMaxsi is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for
|
|
|
|
more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
|
|
along with LibMaxsi. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
stdio.c
|
|
|
|
Sets up stdin, stdout, stderr.
|
|
|
|
|
|
|
|
******************************************************************************/
|
|
|
|
|
2012-02-12 19:07:02 -05:00
|
|
|
#define __SORTIX_STDLIB_REDIRECTS 0
|
Implemented large parts of the stdio(3), including fprintf.
Made FILE an interface to various backends. This allows application writers
to override the standard FILE API functions with their own backends. This
is highly unportable - it'd be nice if a real standard existed for this.
glibc already does something like this internally, but AFAIK you can't hook
into it.
Added fdopen(3), fopen(3), fregister(3), funregister(3), fread(3),
fwrite(3), fseek(3), clearerr(3), ferror(3), feof(3), rewind(3), ftell(3),
fflush(3), fclose(3), fileno(3), fnewline(3), fcloseall(3), memset(3),
stdio(3), vfprintf(3), fprintf(3), and vprintf(3).
Added a file-descriptor backend to the FILE API.
fd's {0, 1, 2} are now initialized as stdin, stdout, and stderr when the
standard library initializes.
fcloseall(3) is now called on exit(3).
decl/intn_t_.h now @include(size_t.h) instead of declaring it itself.
Added <stdint.h>.
The following programs now flush stdout: cat(1), clear(1), editor(1),
init(1), mxsh(1).
printf(3) is now hooked up against vprintf(3), while Maxsi::PrintF
remains using the system call, for now.
2011-12-23 22:08:10 -05:00
|
|
|
#include <stdio.h>
|
2012-02-12 19:07:02 -05:00
|
|
|
#include <stdlib.h>
|
Implemented large parts of the stdio(3), including fprintf.
Made FILE an interface to various backends. This allows application writers
to override the standard FILE API functions with their own backends. This
is highly unportable - it'd be nice if a real standard existed for this.
glibc already does something like this internally, but AFAIK you can't hook
into it.
Added fdopen(3), fopen(3), fregister(3), funregister(3), fread(3),
fwrite(3), fseek(3), clearerr(3), ferror(3), feof(3), rewind(3), ftell(3),
fflush(3), fclose(3), fileno(3), fnewline(3), fcloseall(3), memset(3),
stdio(3), vfprintf(3), fprintf(3), and vprintf(3).
Added a file-descriptor backend to the FILE API.
fd's {0, 1, 2} are now initialized as stdin, stdout, and stderr when the
standard library initializes.
fcloseall(3) is now called on exit(3).
decl/intn_t_.h now @include(size_t.h) instead of declaring it itself.
Added <stdint.h>.
The following programs now flush stdout: cat(1), clear(1), editor(1),
init(1), mxsh(1).
printf(3) is now hooked up against vprintf(3), while Maxsi::PrintF
remains using the system call, for now.
2011-12-23 22:08:10 -05:00
|
|
|
#include <string.h>
|
2012-02-12 19:07:02 -05:00
|
|
|
#include <errno.h>
|
Implemented large parts of the stdio(3), including fprintf.
Made FILE an interface to various backends. This allows application writers
to override the standard FILE API functions with their own backends. This
is highly unportable - it'd be nice if a real standard existed for this.
glibc already does something like this internally, but AFAIK you can't hook
into it.
Added fdopen(3), fopen(3), fregister(3), funregister(3), fread(3),
fwrite(3), fseek(3), clearerr(3), ferror(3), feof(3), rewind(3), ftell(3),
fflush(3), fclose(3), fileno(3), fnewline(3), fcloseall(3), memset(3),
stdio(3), vfprintf(3), fprintf(3), and vprintf(3).
Added a file-descriptor backend to the FILE API.
fd's {0, 1, 2} are now initialized as stdin, stdout, and stderr when the
standard library initializes.
fcloseall(3) is now called on exit(3).
decl/intn_t_.h now @include(size_t.h) instead of declaring it itself.
Added <stdint.h>.
The following programs now flush stdout: cat(1), clear(1), editor(1),
init(1), mxsh(1).
printf(3) is now hooked up against vprintf(3), while Maxsi::PrintF
remains using the system call, for now.
2011-12-23 22:08:10 -05:00
|
|
|
#include "fdio.h"
|
|
|
|
|
|
|
|
FILE* stdin;
|
|
|
|
FILE* stdout;
|
|
|
|
FILE* stderr;
|
|
|
|
|
|
|
|
int init_stdio()
|
|
|
|
{
|
|
|
|
stdin = fdio_newfile(0, "r");
|
|
|
|
stdout = fdio_newfile(1, "w");
|
|
|
|
stderr = fdio_newfile(2, "w");
|
|
|
|
return 0;
|
|
|
|
}
|
2011-12-29 19:06:27 -05:00
|
|
|
|
|
|
|
int getchar(void)
|
|
|
|
{
|
|
|
|
return fgetc(stdin);
|
|
|
|
}
|
|
|
|
|
|
|
|
int putchar(int c)
|
|
|
|
{
|
|
|
|
return fputc(c, stdout);
|
|
|
|
}
|
2012-01-07 19:57:45 -05:00
|
|
|
|
2012-02-12 19:07:02 -05:00
|
|
|
ssize_t getdelim(char** lineptr, size_t* n, int delim, FILE* fp)
|
|
|
|
{
|
|
|
|
if ( !lineptr || (*lineptr && !n) || !fp ) { errno = EINVAL; return -1; }
|
|
|
|
const size_t DEFAULT_BUFSIZE = 32UL;
|
|
|
|
int malloced = !*lineptr;
|
|
|
|
if ( malloced ) { *lineptr = (char*) malloc(DEFAULT_BUFSIZE); }
|
|
|
|
if ( !*lineptr ) { return -1; }
|
|
|
|
size_t bufsize = malloced ? DEFAULT_BUFSIZE : *n;
|
|
|
|
if ( n ) { *n = bufsize; }
|
|
|
|
ssize_t written = 0;
|
|
|
|
int c;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if ( (c = getc(fp)) == EOF ) { goto cleanup; }
|
|
|
|
if ( bufsize <= (size_t) written + 1UL )
|
|
|
|
{
|
|
|
|
size_t newbufsize = 2UL * bufsize;
|
|
|
|
char* newbuf = (char*) realloc(*lineptr, newbufsize);
|
|
|
|
if ( !newbuf ) { goto cleanup; }
|
|
|
|
bufsize = newbufsize;
|
|
|
|
if ( n ) { *n = bufsize; }
|
|
|
|
*lineptr = newbuf;
|
|
|
|
}
|
|
|
|
(*lineptr)[written++] = c;
|
|
|
|
} while ( c != delim );
|
|
|
|
(*lineptr)[written] = 0;
|
|
|
|
return written;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
free(malloced ? *lineptr : NULL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t getline(char** lineptr, size_t* n, FILE* fp)
|
|
|
|
{
|
|
|
|
return getdelim(lineptr, n, '\n', fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
char* sortix_gets(void)
|
|
|
|
{
|
|
|
|
char* buf = NULL;
|
|
|
|
size_t n;
|
|
|
|
if ( getline(&buf, &n, stdin) < 0 ) { return NULL; }
|
|
|
|
size_t linelen = strlen(buf);
|
|
|
|
if ( linelen && buf[linelen-1] == '\n' ) { buf[linelen-1] = 0; }
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2012-01-07 19:57:45 -05:00
|
|
|
int puts(const char* str)
|
|
|
|
{
|
|
|
|
return printf("%s\n", str) < 0 ? EOF : 1;
|
|
|
|
}
|
|
|
|
|