OpenCores
URL https://opencores.org/ocsvn/an-fpga-implementation-of-low-latency-noc-based-mpsoc/an-fpga-implementation-of-low-latency-noc-based-mpsoc/trunk

Subversion Repositories an-fpga-implementation-of-low-latency-noc-based-mpsoc

[/] [an-fpga-implementation-of-low-latency-noc-based-mpsoc/] [trunk/] [mpsoc/] [src_processor/] [src_lib/] [simple-printf/] [sim_fprintf.c] - Rev 48

Compare with Previous | Blame | View Log

/*
 * Copyright (c) 2016, Matt Redfearn
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice,
 * this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * 3. Neither the name of the copyright holder nor the names of its
 * contributors may be used to endorse or promote products derived from this
 * software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */
 
#include <stddef.h>
#include <stdarg.h>
 
#ifdef TEST
#include <stdio.h>
#endif /* TEST */
 
 
 
static void fsimple_outputchar(char  *file_ptr, char c)
{
 
		foutbyte(c);
 
}
 
enum fflags {
	fPAD_ZERO	= 1,
	fPAD_RIGHT	= 2,
};
 
static int fprints(char *out, const char *string, int width, int flags)
{
	int pc = 0, padchar = ' ';
 
	if (width > 0) {
		int len = 0;
		const char *ptr;
		for (ptr = string; *ptr; ++ptr) ++len;
		if (len >= width) width = 0;
		else width -= len;
		if (flags & fPAD_ZERO)
			padchar = '0';
	}
	if (!(flags & fPAD_RIGHT)) {
		for ( ; width > 0; --width) {
			fsimple_outputchar(out, padchar);
			++pc;
		}
	}
	for ( ; *string ; ++string) {
		fsimple_outputchar(out, *string);
		++pc;
	}
	for ( ; width > 0; --width) {
		fsimple_outputchar(out, padchar);
		++pc;
	}
 
	return pc;
}
 
#define PRINT_BUF_LEN 64
 
static int fsimple_outputi(char *out, int i, int base, int sign, int width, int flags, int letbase)
{
	char print_buf[PRINT_BUF_LEN];
	char *s;
	int t, neg = 0, pc = 0;
	unsigned int u = i;
 
	if (i == 0) {
		print_buf[0] = '0';
		print_buf[1] = '\0';
		return fprints(out, print_buf, width, flags);
	}
 
	if (sign && base == 10 && i < 0) {
		neg = 1;
		u = -i;
	}
 
	s = print_buf + PRINT_BUF_LEN-1;
	*s = '\0';
 
	while (u) {
		t = u % base;
		if( t >= 10 )
			t += letbase - '0' - 10;
		*--s = t + '0';
		u /= base;
	}
 
	if (neg) {
		if( width && (flags & fPAD_ZERO) ) {
			fsimple_outputchar (out, '-');
			++pc;
			--width;
		}
		else {
			*--s = '-';
		}
	}
 
	return pc + fprints (out, s, width, flags);
}
 
 
static int fsimple_vsprintf(char *out, char *format, va_list ap)
{
	int width, flags;
	int pc = 0;
	char scr[2];
	union {
		char c;
		char *s;
		int i;
		unsigned int u;
		void *p;
	} u;
 
	for (; *format != 0; ++format) {
		if (*format == '%') {
			++format;
			width = flags = 0;
			if (*format == '\0')
				break;
			if (*format == '%')
				goto out;
			if (*format == '-') {
				++format;
				flags = fPAD_RIGHT;
			}
			while (*format == '0') {
				++format;
				flags |= fPAD_ZERO;
			}
			if (*format == '*') {
				width = va_arg(ap, int);
				format++;
			} else {
				for ( ; *format >= '0' && *format <= '9'; ++format) {
					width *= 10;
					width += *format - '0';
				}
			}
			switch (*format) {
				case('d'):
				case('i'):
					u.i = va_arg(ap, int);
					pc += fsimple_outputi(out, u.i, 10, 1, width, flags, 'a');
					break;
 
				case('u'):
					u.u = va_arg(ap, unsigned int);
					pc += fsimple_outputi(out, u.u, 10, 0, width, flags, 'a');
					break;
 
				case('x'):
					u.u = va_arg(ap, unsigned int);
					pc += fsimple_outputi(out, u.u, 16, 0, width, flags, 'a');
					break;
 
				case('X'):
					u.u = va_arg(ap, unsigned int);
					pc += fsimple_outputi(out, u.u, 16, 0, width, flags, 'A');
					break;
 
				case('c'):
					u.c = va_arg(ap, int);
					scr[0] = u.c;
					scr[1] = '\0';
					pc += fprints(out, scr, width, flags);
					break;
 
				case('s'):
					u.s = va_arg(ap, char *);
					pc += fprints(out, u.s ? u.s : "(null)", width, flags);
					break;
				default:
					break;
			}
		}
		else {
out:
			fsimple_outputchar (out, *format);
			++pc;
		}
	}
	//if (out) **out = '\0';
	return pc;
}
 
 
 
int simple_fprintf(char *buf, char *fmt, ...)
{
	va_list ap;
	int r;
	fout_select(buf);
	va_start(ap, fmt);
	r = fsimple_vsprintf(buf, fmt, ap);
	va_end(ap);
 
	return r;
}
 
 
 
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.