Autor
|
Tema: Implementación de printf no termina con null las cadenas (Leído 3,011 veces)
|
huchoko
|
Buenas, he hecho una implementación de printf, sin usar vsnprintf (por que no puedo). Funciona bien, pero si trato de formatear un número hexadecimal y aveces cadenas, se descontrola todo, por lo que pienso que se me está colando un terminador nulo '\0' . Ejemplo: kputs("r8: %x r9: %x r10: %x", r->r8, r->r9, r->r10);
Salida: Sólo me imprime r8 y de ahí el valor en hexadecimal, pero los demás los ignora y no imprime r9 ni r0. Nótese el carácter "└" el cual no es esperado. int kputs(const char *fmt, ...) { /* Hacer nada si la cadena esta vacia. */ if (!fmt) return 0; char buffer[1024] = {0}; va_list args; size_t cout; unsigned int* charc; for (cout = 0; cout < strlen(fmt ); ++cout ) { switch (fmt[cout]) { case '%': switch (fmt[cout + 1]) { case 'c': charc = va_arg(args , unsigned int); putch(charc); ++cout; break; case 's': charc = (int*) (va_arg(args , unsigned int)); strcpy(buffer , (const char*) charc ); print_string(buffer); ++cout; break; case 'i': print_string(buffer); ++cout; break; case 'X': case 'x': itoa((unsigned)charc , buffer , 16); print_string(buffer); ++cout; case '%': putch("%"); ++cout; default: return 1; } break; default: putch(fmt[cout]); break; } } return cout; }
Las funciones itoa no creo que sean el problema, por que mi implementación no tiene problemas para imprimir enteros. (excepto que se desborda con un número muy grande, eso se arregla fácil)
|
|
« Última modificación: 8 Abril 2019, 03:30 am por hextakatt »
|
En línea
|
|
|
|
CalgaryCorpus
|
Tienes que variar el tipo del segundo parametro de va_arg. Tu solo usas unsigned int (o int) y deberias usar el tipo que calce con el % que se pasa. Mira el ejemplo al final de aqui: https://linux.die.net/man/3/va_arg
|
|
|
En línea
|
|
|
|
huchoko
|
Tienes que variar el tipo del segundo parametro de va_arg. Tu solo usas unsigned int (o int) y deberias usar el tipo que calce con el % que se pasa. Mira el ejemplo al final de aqui: https://linux.die.net/man/3/va_argGracias, pero aún así no me funciona, mismo problema.
|
|
|
En línea
|
|
|
|
|
huchoko
|
Mismo problema, hacer un cast de int a char cuando trato de imprimir hexadecimal no imprime nada, y el mismo problema... int kputs(const char *fmt, ...) { /* Empty string sanity test */ if (!fmt) return 0; char buffer[1024] = {0}; va_list args; size_t cout; unsigned int* charc; for (cout = 0; cout < strlen(fmt ); ++cout ) { switch (fmt[cout]) { case '%': switch (fmt[cout + 1]) { case 'c': charc = (char)va_arg(args , int); putch(charc); ++cout; break; case 's': strcpy(buffer , (const char*) charc ); print_string(buffer); ++cout; break; case 'i': print_string(buffer); ++cout; break; case 'X': case 'x': charc = (char)va_arg(args , int); print_string(buffer); ++cout; case '%': putch("%"); ++cout; default: return 1; } break; default: putch(fmt[cout]); break; } } return cout; }
|
|
|
En línea
|
|
|
|
CalgaryCorpus
|
sugiero que recibas lo que va_arg te retorna en una variable del tipo que pides. Ejemplo en linea 28 y siguientes: cambia por int miVariableInt; // al inicio de la funcion // y luego en el switch, para el caso .'i' miVariableInt = va_arg(args , int); itoa(miVariableInt , buffer , 10); print_string(buffer);
Y si esto se comporta bien, hacer los mismos con los otros tipos.
|
|
|
En línea
|
|
|
|
huchoko
|
Ahora si funciona para los enteros, pero aún está el problema con los hexadecimales. Osea, si le paso el hexadecimal 0xF2, imprime 0xF2, pero no tiene el terminador nulo, lo cual hace que se imprima un caracter aleatorio después de imprimir el hexadecimal. (Sólo pasa cuando se trata de imprimir hexadecimal, los demás están ok)
|
|
« Última modificación: 7 Abril 2019, 22:59 pm por hextakatt »
|
En línea
|
|
|
|
RayR
Desconectado
Mensajes: 243
|
Ese error se debe a que te falta un break para el case 'x' (y ya que estamos, para el case '%'). Hay más problemas, pero el principal es lo que ya te dijeron, que debes usar variables de distintos tipos según el % que recibas. Tal como lo tienes es una suerte que funcione. Dado que char c es un puntero, cuando se ejecuta esto: charc = va_arg(args, int);
Lo que en realidad estás haciendo es que charc apunte a la dirección de memoria cuyo número le enviaste a kputs. Por ejemplo, si le envias 22, charc estará apuntando a la dirección 22. Cuando llamas a itoa, en realidad le estás pasando una dirección de memoria, pero como esta función espera un simple entero, lo interpreta como tal, y por eso no falla. Pero si en algún punto se intentara desreferenciar a charc, tu programa fallaría estrepitosamente , seguramente generando una violación de acceso. charc debería ser un simple int, o unsigned, y para el case 's' debes usar un char *.
|
|
|
En línea
|
|
|
|
huchoko
|
Gracias, funciona perfectamente! Gracias a ambos! Ese error se debe a que te falta un break para el case 'x' (y ya que estamos, para el case '%').
Ah, ¡esos condenados break! siempre me pasa, y por más que reviso el código siempre se me cola uno Saludos
|
|
|
En línea
|
|
|
|
|
Mensajes similares |
|
Asunto |
Iniciado por |
Respuestas |
Vistas |
Último mensaje |
|
|
getc() y printf
Programación C/C++
|
m@o_614
|
1
|
2,136
|
12 Septiembre 2013, 23:44 pm
por ecfisa
|
|
|
Curiosidad con printf en C
Programación C/C++
|
milx86
|
2
|
2,064
|
11 Octubre 2014, 20:46 pm
por milx86
|
|
|
(Ayuda) La función wglMakeCurrent(NULL, NULL) crashea el programa a veces
Programación C/C++
|
Seyro97
|
1
|
2,192
|
10 Junio 2015, 15:34 pm
por Eternal Idol
|
|
|
Problemas con printf();
Programación C/C++
|
soyloqbuskas
|
4
|
3,339
|
20 Octubre 2015, 00:57 am
por soyloqbuskas
|
|
|
[C][?] No se ejecutan printf
Programación C/C++
|
JoseluCross
|
5
|
3,519
|
25 Marzo 2016, 15:07 pm
por JoseluCross
|
|