Solution:
mount -t usbfs none /proc/bus/usb
2011年7月5日
2011年7月4日
udevd: error getting socket: Address family not supported by protocol
Built-in "UNIX DOMAIN SOCKETS". (CONFIG_UNIX)
2011年6月30日
Array and Pointers
--------------------------------------------
Snippet:
char *value="12345";
int size;
size = sizeof(value);
printf("1.value[%s], size[%d]\n", value, size);
strcpy(value, "67890");
printf("2.value[%s], size[%d]\n", value, size);
--------------------------------------------
Result:
1.value[12345], size[4]
Segmentation fault
--------------------------------------------
Snippet:
char value[]="12345";
int size;
size = sizeof(value);
printf("1.value[%s], size[%d]\n", value, size);
strcpy(value, "67890");
printf("2.value[%s], size[%d]\n", value, size);
--------------------------------------------
Result:
1.value[12345], size[6]
2.value[67890], size[6]
標籤: Programming
2011年6月29日
error: linux/autoconf.h: No such file or directory
The kernel header file autoconf.h no longer lives at include/linux/autoconf.h.
It's now at include/generated/autoconf.h.
2011年6月22日
warning: type defaults to 'int' in declaration of 'EXPORT_SYMBOL'
warning: type defaults to 'int' in declaration of 'EXPORT_SYMBOL'
Solution:
Add "#include <linux/module.h>" in your code.
warning: implicit declaration of function 'memset'
warning: implicit declaration of function 'memset'
warning: implicit declaration of function 'open'
Solution:
Add "#include <string.h>" and "#include <unistd.h>" in your code.
error: 'TASK_INTERRUPTIBLE' undeclared (first use in this function)
When moving from v2.6.21 to v2.6.33 kernel, see errors below:
error: 'TASK_INTERRUPTIBLE' undeclared (first use in this function)
Solution:
Add "#include <linux/sched.h> " in your code.
2011年6月21日
warning: incompatible implicit declaration of built-in function 'malloc'
warning: incompatible implicit declaration of built-in function 'malloc'
Solution:
Add "#include <stdlib.h>" in your code.
2011年5月30日
Get external defined string from Makefile
gcc -D 'HEADER_PREFIX="MYIMAGE"' ...
#define IMAGE_HEADER HEADER_PREFIX"_PKG"
printf("My image full header [%s]\n", IMAGE_HEADER);
標籤: Programming
2011年5月18日
IPv6 forwarding seems to be disabled, exiting
echo "1" > /proc/sys/net/ipv6/conf/all/forwarding
or
sysctl net.ipv6.conf.all.forwarding=1
2011年5月6日
IPv6 relevance commands
# /sbin/ip -6 addr add $ipv6address/$prefixlength dev $interface
# /sbin/ifconfig $interface inet6 add $ipv6address/$prefixlength
# /sbin/ip -6 route show [dev $device]
# /sbin/ip -6 route add $ipv6network/$prefixlength via $ipv6address [dev $device]
# /sbin/ip -6 route del $ipv6network/$prefixlength via $ipv6address [dev $device]
# /sbin/ip -6 route add 2000::/3 dev eth0 metric 1
# /sbin/ip -6 neigh show
# /sbin/ip neighbor
2011年4月27日
cross-compile:libtool
From: http://orzlab.blogspot.com/2007/05/cross-compilelibtool.html
/bin/sh libtool --mode=link target-gcc -c -O2 -o libbar.so ... -lfoo
target-gcc -c -O2 -o libbar.so ... /usr/lib/libfoo.so
/usr/lib/libfoo.so: could not read symbols: File in wrong format
collect2: ld returned 1 exit status
In cross-compilation, the link should point to target installation path, not local host path.
Solution:
1. Edit libfoo.la.
2. Modify libdir='/usr/lib' to target lib installation path. Such as "/export/arm_toolchain/usr/lib".
2011年4月12日
system() return (10): No child processes
The code includes "signal(SIGCHLD, SIG_IGN); " to prevent process from zombie.
So the wait() is unable to get the signal and system() will return -1.
Solution:
Change from SIG_IGN to SIG_DFL.
In my case, I have to retain signal(SIGCHLD, SIG_IGN), so how to handle well with its return value?
標籤: Programming
2011年4月7日
popen hangs!?
The suspicion is waitpid()?
After comment it out, the code works well and doesn't hang on popen().
Ref: http://pubs.opengroup.org/onlinepubs/009695399/functions/pclose.html
The issue may be related to pclose().
The return value of pclose() is always "-1", and the next popen() call hangs then.
From functional description, it says:
The pclose() function shall close a stream that was opened by popen(), wait for the command to terminate, and return the termination status of the process that was running the command language interpreter. However, if a call caused the termination status to be unavailable to pclose(), then pclose() shall return -1 with errno set to [ECHILD] to report this situation. This can happen if the application calls one of the following functions:
- wait()
- waitpid() with a pid argument less than or equal to 0 or equal to the process ID of the command line interpreter
- Any other function not defined in this volume of IEEE Std 1003.1-2001 that could do one of the above.
In any case, pclose() shall not return before the child process created by popen() has terminated.
And in my case, the signal handler routine is using waitpid(-1, &stat, WNOHANG). After change it to waitpid(forked_pid, &stat, WNOHANG) to ignore SIGCHLD triggered by popen(), the application doesn't hang anymore.
標籤: Programming
2011年4月6日
time.h: undefined reference to `__aeabi_uldivmod'
linux_2_6_21/include/linux/time.h:184: undefined reference to `__aeabi_uldivmod'
Kernel: v2.6.21.1
Compiler: gcc v4.3.3
Solution:
From http://lkml.org/lkml/2008/2/22/464.
--- include/linux/time.h 2008-10-09 19:47:23.000000000 +0200
+++ include/linux/time.h.new 2008-10-09 19:47:54.000000000 +0200
@@ -173,6 +173,10 @@
{
ns += a->tv_nsec;
while(unlikely(ns >= NSEC_PER_SEC)) {
+ /* The following asm() prevents the compiler from
+ * optimising this loop into a modulo operation. */
+ asm("" : "+r"(ns));
+
ns -= NSEC_PER_SEC;
a->tv_sec++;
}
2011年3月21日
Hack svnmailer: Subject with "Log"
Change commit email subject to be the first line of change log.
Fedora 14
svnmailer-1.0.8
1. vi /usr/lib/python2.7/site-packages/svnmailer/notifier/_mail.py
2. Add the following blue line at L:334.
subject = dosubject('files')
if len(subject) > short_length:
subject = dosubject('dirs')
subject = "%s %s" % (subject, self.getLog())
# reduce to the max ...
if max_length and len(subject) > max_length:
subject = "%s..." % subject[:max_length - 3]
2011年3月15日
Linux Kernel and Floating Point
From: http://www.linuxsmiths.com/blog/?p=253
Consider the following kernel module code snippet that does a floating point divide. (The complete module code is here).
static noinline double dummy_float_divide(double arg1, double arg2)
{
return (arg1 / arg2);
}
When we compile the given module, we get.
CC [M] /home/lsmiths/linux_kernel_fp_support/fp.o
Building modules, stage 2.
MODPOST 1 modules
WARNING: "__divdf3" [/home/lsmiths/linux_kernel_fp_support/fp.ko] undefined!
CC /home/lsmiths/linux_kernel_fp_support/fp.mod.o
LD [M] /home/lsmiths/linux_kernel_fp_support/fp.ko
An attempt to load this module fails with the following error in dmesg.
fp: Unknown symbol __divdf3
__divdf3 sounds like a function for dividing floating point numbers, but how did it make it to our module. We never used it!
Disassmbling the module object (objdump -d fp.o), yields
00000000
0: 55 push %ebp
1: 89 e5 mov %esp,%ebp
3: 83 ec 10 sub $0x10,%esp
6: 8b 45 10 mov 0x10(%ebp),%eax
9: 8b 55 14 mov 0x14(%ebp),%edx
c: 89 44 24 08 mov %eax,0x8(%esp)
10: 8b 45 08 mov 0x8(%ebp),%eax
13: 89 54 24 0c mov %edx,0xc(%esp)
17: 8b 55 0c mov 0xc(%ebp),%edx
1a: 89 04 24 mov %eax,(%esp)
1d: 89 54 24 04 mov %edx,0x4(%esp)
21: e8 fc ff ff ff call 22
26: c9 leave
27: c3 ret
Lets confirm it by looking at the relocation table readelf -r fp.o output has the following.
Relocation section '.rel.text' at offset 0x3ee0 contains 1 entries:
Offset Info Type Sym.Value Sym. Name
00000022 00001902 R_386_PC32 00000000 __divdf3 <-- The assembly code above references this function
So what is happening is that, gcc replaced the expression (arg1/arg2) by a call to __divdf3 function which is supposed to carry out the floating point division using integer arithmetic. Why did gcc do that ? and not generate actual assembly instructions to do floating point divide.
This is because, the module code was compiled with -msoft-float gcc option, which instructs gcc to not generate floating point assembly instructions and instead generate calls to the glibc s/w floating point emulation functions. -msoft-float is useful when compiling programs for platforms that do not have hardware floating point support. Nothing is wrong with this. Infact if you compile an equivalent user program with -msoft-float, it should work (pls read note below)
P.S. Actually it depends on whether your glibc is compiled with software floating point emulation support. Usually x86 based default glibc distributions come w/o soft floating point emulation, as almost all x86 platforms have h/w floating point support. If h/w floating point support is present, it is preferred because of its speed and the fact that it puts less load on the CPU (for applications with extensive floating point usage, f.e. sone gaming applications or CAD design applications etc). You can use the following command to see if your glibc distribution has software floating point emulation support
# ldd /bin/ls | grep libc | awk '{print $3}' | xargs readlink -f | xargs nm -D | grep __divdf3
I believe the reason why default glibc does not come with soft float support enabled, is to prevent applications from accidentally using soft float. Otherwise if some application is unintentionally compiled with -msoft-float, the user will never know and the application will be using the inefficient soft float, even though h/w float support is available :-(
So, till now we know the following things :
1.Linux kernel (and all its modules) are compiled with -msoft-float gcc option (to know why, read on)
2.Linux kernel (and all its modules) are _not_ linked with glibc and hence we do not have access to soft floating point emulation functions (like __divdf3).
3.Linux kernel itself does not provide its own implementation of __divdf3 (and other soft floating point functions).
The above explains why we get the error while compiling and loading the module, but the inquisitive of us will still be having few questions. Lets try to find answers to those questions.
What is floating point and how is it handled ?
Before we get into the main topic of the discussion, i.e. the state of floating point support in Linux, and the reasons behind that, lets take a quick look at what it takes to support floating point operations.
Floating point usage is not very common. So much so that x86 designers did not make the floating point unit (the CPU real estate needed for floating point operations) part of the original CPU. In-fact floating point instructions were supported by a special coprocessor. For the 8086 this was called 8087. Similarly for other 80×86 processors the corresponding floating point coprocessor was called 80×87. Till the 80386 processor, this coprocessor came as a separate chip which used to sit alongside the main CPU and all the floating point calculations were directed to it, which then could use its floating point unit (FPU) to do the calculations and pass the result back to the main processor. Starting 80486, the FPU was integrated with the main CPU, but still the FPU was a logically separate unit, i.e. it used a separate register set to load/store the floating point values and it used a different ALU for carrying out the floating point calculations.
The reason for keeping the FPU separate is twofold.
1.floating point operations are very rarely used, and
2.floating point operations are expensive.
This design has a very important impact on how floating point is handled in the present operating systems. Had the floating point support been native to a processor, just like the integer support, then it would not be treated any differently and we would use them just like we use the integer operations. This blog would not exist !
In this article, wherever necessary, we will take the x87 FPU as an example, but all this should apply to any other processor and its corresponding FPU.
Other ways of handling floating point
What we just discussed above is called the hardware floating point support, as the floating point operations are handled in the hardware. Since the FPU is separate from the processor there is a possibility that we do not have the FPU in a certain system. Note that this does not apply to modern x86 based systems since FPU comes on the same die as the main processor, so if you buy the processor, you get the FPU also. Other architectures, especially those used for embedded system design, might still make the FPU as an add-on for cost reasons. In such cases, where the FPU is not present in the system and we need to still do a few floating point calculations, we have the following options.
Do not use floating point
Instead use the fixed point arithmetic using integer operations. This can be used if our floating point usage is not much and we do not need very high precision. Also, every application does it in its own way leading to lots of inconsitencies and possible errors.
Use a floating point emulation library
The application program written in high-level language uses the floating point operations as-is, but the compiler, instead of generating floating point instructions for them, generates calls to the floating point emulation functions. These emulation functions are provided by some library, against which the program is then linked. The GNU C Library glibc also comes with support for floating point emulation. Note that the default glibc distribution might not have the floating point emultion (FPE) support, but glibc has a configure option using which we can compile glibc with FPE support. This needs support from the compiler, as it has to identify floating point operations and generate FPE calls for them. Usually compilers provide some commandline option for this. gcc provides the -msoft-float option for this purpose. This is not the default and w/o this option gcc generates floating point instructions.
Kernel floating point emulation
If we need to emulate floating point operations and we want to hide it from the applications, we can have the kernel emulate them. This can be kept completely transparent from the applications and they won’t even know if the underlying processor has a h/w FPU or not, but for the slowness that it might cause.
This is implemented by the CPU generating an exception every time it encounters a floating point instruction, and the kernel exception handler then emulating the instruction using integer arithmetic.
For this we need support from the CPU, i.e. it should generate an exception on encountering a floating point instruction. x86 processors provide this support by means of an Emulation bit (bit #2 in CR0 register). If the h/w FPU is not present then this bit will be set. When the Emulation bit (abbreviated as EM) is set, the x86 CPU will raise the Device Not Available (#NM) exception every single time it encounters a floating point instruction. A Linux kernel compiled with floating point emulation support, will then handle the emulation inside the exception handler, and the application will run seamlessly. If the Linux kernel is not compiled with FPE support, it raises SIGFPE to the application.
The Floating Point Context
Floating point unit, uses its own set of registers for doing the floating point arithmetic, f.e. the x87 FPU (coprocessor unit for x86 processor) uses the following registers for floating point arithmetic
8 data registers (ST0-ST7) The status register The control register The tag word register The last instruction point register Last data (operand) pointer register Opcode register
These are registers used specifically for floating point arithmetic and are completely separate from the native x86 registers used for integer arithmetic. These constitute the floating point context of the CPU. This (apart from the native processor context) need to be saved/restored with each process context switch. This seems like a big price to pay :-(
Cheer up !
we have a smart way to handle this. Read on …
Because floating point usage is not very common (infact many times a process will not execute any floating point instruction in its whole quantum) and because floating point registers are so large and plentiful, it does not make sense to save and restore floating point registers on every context switch. Most of the times this save/restore effort will be wasted, as the registers would not have been dirtied. x86 designers were smart enough to think about this beforehand and hence they added a bit in the CR0 register which can be used by the operating system to do this save/restore efficiently, i.e. floating point registers are saved at context switch out time, only if the going-out process executed some floating point instruction in that quantum, hence modifying the CPU FP registers. Similarly, the floating point registers are restored only when the process wants to execute some floating point instruction, hence needing the FP registers.
I was referring to the Task Switched bit (bit #3) in the CR0 register. As the name implies, the processor sets this bit on every task switch. Pls note that since Linux does not use the CPU provided task switching facility, but instead does the task switch by hand, Linux has to set the TS bit explicitly as part of the task switch. Irrespective of how the TS bit is set, its significance is that when this bit is set the CPU generates a Device Not Available (#NM) exception, when a floating point instruction is executed (for the TS bit to have effect the EM bit should be cleared, else irrespective of the TS bit the CPU raises the #NM exception for every floating point instruction). This one feature provided by the CPU can be used by the OS to do efficient context switches involving floating point context.
How ?
Lets look at how Linux uses this to do intelligent save and restore of FP registers. Lets first see how and when is the TS bit set in the CR0 register, since if the TS bit is not set, the Device Not Available (#NM) exception will not be generated and we won’t be notified of floating point instruction execution. The TS bit is set from cpu_init() initially, so that the first process that runs a floating point instruction causes the Device Not Available (#NM) exception. TS bit is then cleared from the Device Not Available (#NM) exception handler, so that no further floating point instructions executing from the current process, in its current quantum, cause the Device Not Available (#NM) exception.
The TS bit is then set again when the current process is scheduled out. so that the new process executing a floating point instruction also causes the Device Not Available (#NM) exception. This is done from the context switch-out path — __switch_to()->__unlazy_fpu().
In short, the Linux kernel wants to be notified of (and only) the first floating point instruction that a process executes in a quantum. It then takes appropriate action to restore the floating point state of that process. This ensures that the floating point state of a process is restored (i.e. saved FP state of the process loaded on to the CPU FP state) only (and only) when the process executes at least one floating point instruction. If a process does not execute any floating point instruction in a certain quantum, there is no need to restore the saved floating point state of that process. Also, since the saved FPU state of the current process did not change, we need not save the FPU state when this process is switched out. In such case the CPU FPU state remains the same as it was before the current process started running and if that corresponds to the next-to-run process’ FPU state, we need not even restore its FPU state, as the CPU’s FPU is already has that state. What this means is that if a process does not execute any floating point instruction in a certain quantum, we neither restore nor save the floating point context of that process. So we incur the FP context save/restore overhead when really required :-)
Kernel function math_state_restore() is at the heart of all this. It is called from the Device Not Available (#NM) exception handler, which as we saw before, is called when the TS bit is set and some floating point instruction is executed.
asmlinkage void math_state_restore(void)
{
...
clts(); // we do not want to be called again in this process quantum
/*
* Now that we are going to use the FPU load this process' FPU state in the FPU
*/
if (unlikely(restore_fpu_checking(tsk))) {
stts();
force_sig(SIGSEGV, tsk);
return;
}
thread->status |= TS_USEDFPU; // so that __switch_to->unlazy_fpu can save the FP state of this process
...
}
clts() clears the TS bit as we do not want to be called for all floating point instructions, just the first one. It marks the TS_USEDFPU bit in the current process’ thread->status field. This bit is later checked by the context switch-out code to decide whether to save the FP registers as part of the scheduled out task’s context. Thus Linux kernel ensures that it saves the FP context for a process only if that process executes at least one floating point instruction in its last quantum, hence changing its already saved FPU state. This is the conditional save.
This is about the save optimization. The restore optimization is also present in the math_state_restore() function shown above. Note that, unlike other integer registers, we do not restore the FPU state unconditionally from the context switch-in code. Instead the FP restore is done from the math_state_restore() function, which signifies that the process has executed some floating point instruction, and hence it is necessary to restore the FP state of the process. As we see the floating point state is restored not at the context switch-in time. but just before the process is going to use the floating point state. This is called the lazy restore.
Using floating point in kernel
We learnt how Linux uses conditional save and lazy restore techniques to allow application programs to use the hardware floating point support while avoiding the unnecessary overhead of saving/restoring the FP context on every context switch (even when not required). The assumption in the above discussion is that the only way the FP state of the CPU can change is by the application executing floating point instructions. It assumes that the kernel code will not modify the FP state of the CPU. This effectively means that the kernel code cannot use floating point instructions.
Well.. to be more precise, we cannot use floating point operations in the kernel just like that. We have to follow some discipline. The good news is that the Linux kernel developers have made it very easy to use floating point operations inside the kernel. You just need to surround the floating point code with kernel_fpu_begin() and kernel_fpu_end() and you can safely use floating point operations in the kernel code.
So what magic do these two functions do. Note how the Linux kernel had solved the problem of avoiding unneeded save and restore of FP context when scheduling in/out the user processes. In short the Linux kernel does the following
It sets the TS bit in the CR0 register, before a new process can start execution. This is so that the CPU raises the Device Not Available (#NM) exception when that process runs its first floating point instruction. The kernel can then do the lazy restore of the floating point context of the process.
From the Device Not Available (#NM) exception handler, it sets the TS_USEDFPU flag in the thread->status field. This can then be used by the context switch out code to conditionally save the floating point state of this process.
If we treat the kernel mode also like another process (i.e. something that is capable of changing the FP state of the CPU), we can extend the above logic to allow kernel to use floating point operations safely.
This is exactly what kernel_fpu_begin() and kernel_fpu_end() do.
static inline void kernel_fpu_begin(void)
{
struct thread_info *me = current_thread_info();
preempt_disable();
if (me->status & TS_USEDFPU)
__save_init_fpu(me->task);
else
clts();
}
static inline void kernel_fpu_end(void)
{
stts();
preempt_enable();
}
So if you want to use some floating point operations in the kernel, which can change the FP state of the CPU, we need to first save the FP state of the current process (__save_init_fpu() does that), but only if the current process was doing some floating point operations (me->status & TS_USEDFPU). Then we need to clear the TS bit, so that the CPU does not raise the Device Not Available (#NM) exception anymore.
Once the kernel is done with the floating point operations, it can call kernel_fpu_end() which again sets the TS bit. This causes the Device Not Available (#NM) exception when a new process runs some floating point operations and hence we need to restore its floating point state (since the kernel modified the CPU FP state).
kernel_fpu_begin() and kernel_fpu_end() make sense only if you are using the hardware floating point support in the kernel. For this you will have to compile the kernel (or the module) with -mhard-float option.
One more important thing to keep in mind is that while we are inside kernel_fpu_begin() and kernel_fpu_end() we should not sleep. This is because while we are modifying the CPU FP state, we do not want anyother context to use that FP state.
glibc vs uClibc
From: ftp://ftp.kernel.org/pub/linux/libs/uclibc/Glibc_vs_uClibc_Differences.txt
uClibc and Glibc are not the same — there are a number of differences which may or may not cause you problems. This document attempts to list these differences and, when completed, will contain a full list of all relevant differences. http://loosky.com
1) uClibc is smaller than glibc. We attempt to maintain a glibc compatible interface, allowing applications that compile with glibc to easily compile with uClibc. However, we do not include _everything_ that glibc includes, and therefore some applications may not compile. If this happens to you, please report the failure to the uclibc mailing list, with detailed error messages.
2) uClibc is much more configurable then glibc. This means that a developer may have compiled uClibc in such a way that significant amounts of functionality have been omitted.
3) uClibc does not even attempt to ensure binary compatibility across releases.When a new version of uClibc is released, you may or may not need to recompile all your binaries.
4) malloc(0) in glibc returns a valid pointer to something(!?!?) while in uClibc calling malloc(0) returns a NULL. The behavior of malloc(0) is listed as implementation-defined by SuSv3, so both libraries are equally correct.This difference also applies to realloc(NULL, 0). I personally feel glibc’s behavior is not particularly safe. To enable glibc behavior, one has to explicitly enable the MALLOC_GLIBC_COMPAT option.
4.1) glibc’s malloc() implementation has behavior that is tunable via the MALLOC_CHECK_ environment variable. This is primarily used to provide extra malloc debugging features. These extended malloc debugging features are not available within uClibc. There are many good malloc debugging libraries available for Linux (dmalloc, electric fence, valgrind, etc) that work much better than the glibc extended malloc debugging. So our omitting this functionality from uClibc is not a great loss.
5) uClibc does not provide a database library (libdb). http://loosky.net
6) uClibc does not support NSS (/lib/libnss_*), which allows glibc to easily support various methods of authentication and DNS resolution. uClibc only supports flat password files and shadow password files for storing authentication information. If you need something more complex than this,you can compile and install pam.
7) uClibc’s libresolv is only a stub. Some, but not all of the functionality provided by glibc’s libresolv is provided internal to uClibc. Other functions
are not at all implemented.
8) libnsl provides support for Network Information Service (NIS) which was originally called "Yellow Pages" or "YP", which is an extension of RPC invented
by Sun to share Unix password files over the network. I personally think NIS is an evil abomination and should not be used. These days, using ldap is much
more effective mechanism for doing the same thing. uClibc provides a stub libnsl, but has no actual support for Network Information Service (NIS).
We therefore, also do not provide any of the headers files provided by glibc under /usr/include/rpcsvc.
9) uClibc’s locale support is not 100% complete yet. We are working on it.
10) uClibc’s math library only supports long double as inlines, and even then the long double support is quite limited. Also, very few of the
float math functions are implemented. Stick with double and you should be just fine.
11) uClibc’s libcrypt does not support the reentrant crypt_r, setkey_r and encrypt_r, since these are not required by SuSv3.
12) uClibc directly uses kernel types to define most opaque data types.
13) uClibc directly uses the linux kernel’s arch specific ‘stuct stat’.
14) uClibc’s librt library currently lacks all aio routines, all clock routines, and all shm routines (only the timer routines and the mq routines are implemented).
************* Manuel's Notes *************
Some general comments...
The intended target for all my uClibc code is ANSI/ISO C99 and SUSv3
compliance. While some glibc extensions are present, many will eventually
be configurable. Also, even when present, the glibc-like extensions may
differ slightly or be more restrictive than the native glibc counterparts.
They are primarily meant to be porting _aides_ and not necessarily
drop-in replacements.
Now for some details...
time functions
--------------
1) Leap seconds are not supported.
2) /etc/timezone and the whole zoneinfo directory tree are not supported.
To set the timezone, set the TZ environment variable as specified in
http://www.opengroup.org/onlinepubs/007904975/basedefs/xbd_chap08.html
or you may also create an /etc/TZ file of a single line, ending with a
newline, containing the TZ setting. For example
echo CST6CDT > /etc/TZ
3) Currently, locale specific eras and alternate digits are not supported.
They are on my TODO list.
wide char support
-----------------
1) The only multibyte encoding currently supported is UTF-8. The various
ISO-8859-* encodings are (optionally) supported. The internal
representation of wchar's is assumed to be 31 bit unicode values in
native endian representation. Also, the underlying char encoding is
assumed to match ASCII in the range 0-0x7f.
2) In the next iteration of locale support, I plan to add support for
(at least some) other multibyte encodings.
locale support
--------------
1) The target for support is SUSv3 locale functionality. While nl_langinfo
has been extended, similar to glibc, it only returns values for related
locale entries.
2) Currently, all SUSv3 libc locale functionality should be implemented
except for wcsftime and collating item support in regex.
stdio
-----
1) Conversion of large magnitude floating-point values by printf suffers a loss
of precision due to the algorithm used.
2) uClibc's printf is much stricter than glibcs, especially regarding positional
args. The entire format string is parsed first and an error is returned if
a problem is detected. In locales other than C, the format string is checked
to be a valid multibyte sequence as well. Also, currently at most 10 positional
args are allowed (although this is configurable).
3) BUFSIZ is configurable, but no attempt is made at automatic tuning of internal
buffer sizes for stdio streams. In fact, the stdio code in general sacrifices
sophistication/performace for minimal size.
4) uClibc allows glibc-like custom printf functions. However, while not
currently checked, the specifier must be <= 0x7f.
5) uClibc allows glibc-like custom streams. However, no in-buffer seeking is
done.
6) The functions fcloseall() and __fpending() can behave differently than their
glibc counterparts.
7) uClibc's setvbuf is more restrictive about when it can be called than glibc's
is. The standards specify that setvbuf must occur before any other operations
take place on the stream.
8) Right now, %m is not handled properly by printf when the format uses positional
args.
9) The FILEs created by glibc's fmemopen(), open_memstream(), and fopencookie()
are not capable of wide orientation. The corresponding uClibc routines do
not have this limitation.
10) For scanf, the C99 standard states "The fscanf function returns the value of
the macro EOF if an input failure occurs before any conversion." But glibc's
scanf does not respect conversions for which assignment was surpressed, even
though the standard states that the value is converted but not stored.
glibc bugs that Ulrich Drepper has refused to acknowledge or comment on
( http://sources.redhat.com/ml/libc-alpha/2003-09/ )
-----------------------------------------------------------------------
1) The C99 standard says that for printf, a %s conversion makes no special
provisions for multibyte characters. SUSv3 is even more clear, stating
that bytes are written and a specified precision is in bytes. Yet glibc
treats the arg as a multibyte string when a precision is specified and
not otherwise.
2) Both C99 and C89 state that the %c conversion for scanf reads the exact
number of bytes specified by the optional field width (or 1 if not specified).
uClibc complies with the standard. There is an argument that perhaps the
specified width should be treated as an upper bound, based on some historical
use. However, such behavior should be mentioned in the Conformance document.
3) glibc's scanf is broken regarding some numeric patterns. Some invalid
strings are accepted as valid ("0x.p", "1e", digit grouped strings).
In spite of my posting examples clearly illustrating the bugs, they remain
unacknowledged by the glibc developers.
4) glibc's scanf seems to require a 'p' exponent for hexadecimal float strings.
According to the standard, this is optional.
5) C99 requires that once an EOF is encountered, the stream should be treated
as if at end-of-file even if more data becomes available. Further reading
can be attempted by clearing the EOF flag though, via clearerr() or a file
positioning function. For details concerning the original change, see
Defect Report #141. glibc is currently non-compliant, and the developers
did not comment when I asked for their official position on this issue.
6) glibc's collation routines and/or localedef are broken regarding implicit
and explicit UNDEFINED rules.
More to follow as I think of it...
Profiling:
-------------------------------------------------------------------
uClibc no longer supports 'gcc -fprofile-arcs -pg' style profiling, which
causes your application to generate a 'gmon.out' file that can then be analyzed
by 'gprof'. Not only does this require explicit extra support in uClibc, it
requires that you rebuild everything with profiling support. There is both a
size and performance penalty to profiling your applications this way, as well
as Heisenberg effects, where the act of measuring changes what is measured.
There exist a number of less invasive alternatives that do not require you to
specially instrument your application, and recompile and relink everything.
The OProfile system-wide profiler is an excellent alternative:
http://oprofile.sourceforge.net/
Many people have had good results using the combination of Valgrind
to generate profiling information and KCachegrind for analysis:
http://developer.kde.org/~sewardj/
http://kcachegrind.sourceforge.net/
Prospect is another alternative based on OProfile:
http://prospect.sourceforge.net/
And the Linux Trace Toolkit (LTT) is also a fine tool:
http://www.opersys.com/LTT/
FunctionCheck:
http://www710.univ-lyon1.fr/~yperret/fnccheck/