I am attempting to automatically register data types against a factory. The first non-generalized version worked, but cannot define the registration macro since the factory was changed to a variadic. The compile error is...
error C4430 missing type specifier - int assumed. Not: C++ does
not support default-int.
#define GH_FACTORY_GET_LICENSE_FACTORY \
Factory<ILicenseManager, bool>::getFactory();
#define GH_FACTORY_REGISTER_LICENSE_MANAGER( ) \
Factory<ILicenseManager, bool>::getFactory().registerType();
Example 1
class A
{
public:
A()
{
Factory<gh::license::ILicenseManager, bool>::getFactory().registerType();
}
~A() {}
};
static class A foo;
Example 2
GH_FACTORY_REGISTER_LICENSE_MANAGER;
When using
GH_FACTORY_GET_LICENSE_FACTORY
and
GH_FACTORY_REGISTER_LICENSE_MANAGER
in local scope it builds, but when using
GH_FACTORY_REGISTER_LICENSE_MANAGER
in global scope it doesn't build (looks like the class is undefined?). Any ideas? Would love to understand why this happens.
v9.6, win server 2012, vs 2015
successfully compiled and linked as x64.
Create function fails saying that there is no 'add_one' function in the dll.
postgres=# create function add_one(integer) returns integer as
'win32project1',' add_one' language c strict;
ERROR: could not find function "add_one" in file "C:/Program Files/PostgreSQL/9.6/lib/win32project1.dll"
The function seems to be there though, dumpbin says
1 0 000112CB Pg_magic_func = #ILT+710(Pg_magic_func)
2 1 00011087 pg_finfo_add_one = #ILT+130(pg_finfo_add_one)
3 2 00011190 pg_finfo_add_one_float8 = #ILT+395(pg_finfo_add_one_float8)
4 3 000110F5 pg_finfo_concat_text = #ILT+240(pg_finfo_concat_text)
5 4 000112C1 pg_finfo_copytext = #ILT+700(pg_finfo_copytext)
6 5 0001107D pg_finfo_makepoint = #ILT+120(pg_finfo_makepoint)
OK, there needs to be 2 functions exported for each function, the meta data function pg_finfo_xxx plus the actual function itself xxx.
The standard headers for pg functions compiling mark the meta data function with PGDLLEXPORT but the forward declaration of the actual function is not marked that way. I dont see how this could ever work.
#define PG_FUNCTION_INFO_V1(funcname) \
Datum funcname(PG_FUNCTION_ARGS); \
extern PGDLLEXPORT const Pg_finfo_record * CppConcat(pg_finfo_,funcname)(void); \
const Pg_finfo_record * \
CppConcat(pg_finfo_,funcname) (void) \
{ \
static const Pg_finfo_record my_finfo = { 1 }; \
return &my_finfo; \
} \
extern int no_such_variable
But I made it work by doing
#define PG_FUNCTION_INFO_V1(funcname) \
PGDLLEXPORT Datum funcname(PG_FUNCTION_ARGS); \
extern PGDLLEXPORT const Pg_finfo_record * CppConcat(pg_finfo_,funcname)(void); \
const Pg_finfo_record * \
CppConcat(pg_finfo_,funcname) (void) \
{ \
static const Pg_finfo_record my_finfo = { 1 }; \
return &my_finfo; \
} \
extern int no_such_variable
You didn't show your source file, but it should contain this:
extern PGDLLEXPORT Datum add_one(PG_FUNCTION_ARGS);
PG_FUNCTION_INFO_V1(add_one);
Then it will probably work.
I have started a discussion on the hackes mailing list to implement your answer in PostgreSQL, but it seems that with the build process that PostgreSQL is using (generate and use export definition files) this causes at least warnings, so we have given up on it.
The other option you have is to create and use an export definition file like PostgreSQL does, then you can do without the PGDLLEXPORT decorations altogether.
I want to use SPI1 on a 18F series PIC as master and then use SPI2 on the same PIC as a slave to another PIC ? Can this be done ?
Yes it can. In fact there would be little point providing two serial modules if you were then constrained to use them in the same way.
As it says in the datasheet:
Note that each operates independently from the other. Also, for each module in SPI mode:
You can see that both master and slave modes are supported.
Each MSSP module has its own set of registers, which are used to configure each module as required.
I use the AD7793 as well, but I've rolled my own SPI_READ and SPI_WRITE macros. These macros work on 16-bit processors, and maybe on 32-bit (I have not tried). The first parameter passed is the SPI port number (e.g. 1 or 2). They work in slave and master mode.
e.g. SPI_WRITE(1, 'H'); or SPI_READ(2, char_to_send, char_to_receive_data);
#define _SPI_WRITE_final(_SPIBUF, _SPIRBF, _SPIROV, ch) \
_SPIROV = 0; \
while (_SPIRBF) \
__asm__ volatile ("mov %0, w0" : : "g"(_SPIBUF) : "cc", "w0"); \
_SPIBUF = (uint8_t)(ch); \
__asm__ volatile ("nop"); \
while (!_SPIRBF);
#define _SPI_WRITE_int(bus, ch) \
_SPI_WRITE_final(SPI##bus##BUF, SPI##bus##STATbits.SPIRBF, \
SPI##bus##STATbits.SPIROV, ch)
#define _SPI_WRITE(bus, ch) do {_SPI_WRITE_int(bus, ch);} while(0)
#define _SPI_READ_final(_SPIBUF, _SPIRBF, _SPIROV, ch_out, ch_in) \
_SPIROV = 0; \
while (_SPIRBF) \
__asm__ volatile ("mov %0, w0" : : "g"(_SPIBUF) : "cc", "w0"); \
_SPIBUF = (uint8_t)(ch_out); \
__asm__ volatile ("nop"); \
while (!_SPIRBF); \
ch_in = _SPIBUF;
#define _SPI_READ_int(bus, ch_out, ch_in) \
_SPI_READ_final(SPI##bus##BUF, SPI##bus##STATbits.SPIRBF, \
SPI##bus##STATbits.SPIROV, ch_out, ch_in)
#define _SPI_READ(bus, ch_out, ch_in) do {_SPI_READ_int(bus, ch_out, ch_in);} while(0)
In drivers I often see these three types of init functions being used.
module_init()
core_initcall()
early_initcall()
Under what circumstances should I use them?
Also, are there any other ways of init?
They determine the initialization order of built-in modules. Drivers will use device_initcall (or module_init; see below) most of the time. Early initialization (early_initcall) is normally used by architecture-specific code to initialize hardware subsystems (power management, DMAs, etc.) before any real driver gets initialized.
Technical stuff for understanding below
Look at init/main.c. After a few architecture-specific initialization done by code in arch/<arch>/boot and arch/<arch>/kernel, the portable start_kernel function will be called. Eventually, in the same file, do_basic_setup is called:
/*
* Ok, the machine is now initialized. None of the devices
* have been touched yet, but the CPU subsystem is up and
* running, and memory and process management works.
*
* Now we can finally start doing some real work..
*/
static void __init do_basic_setup(void)
{
cpuset_init_smp();
usermodehelper_init();
shmem_init();
driver_init();
init_irq_proc();
do_ctors();
usermodehelper_enable();
do_initcalls();
}
which ends with a call to do_initcalls:
static initcall_t *initcall_levels[] __initdata = {
__initcall0_start,
__initcall1_start,
__initcall2_start,
__initcall3_start,
__initcall4_start,
__initcall5_start,
__initcall6_start,
__initcall7_start,
__initcall_end,
};
/* Keep these in sync with initcalls in include/linux/init.h */
static char *initcall_level_names[] __initdata = {
"early",
"core",
"postcore",
"arch",
"subsys",
"fs",
"device",
"late",
};
static void __init do_initcall_level(int level)
{
extern const struct kernel_param __start___param[], __stop___param[];
initcall_t *fn;
strcpy(static_command_line, saved_command_line);
parse_args(initcall_level_names[level],
static_command_line, __start___param,
__stop___param - __start___param,
level, level,
&repair_env_string);
for (fn = initcall_levels[level]; fn < initcall_levels[level+1]; fn++)
do_one_initcall(*fn);
}
static void __init do_initcalls(void)
{
int level;
for (level = 0; level < ARRAY_SIZE(initcall_levels) - 1; level++)
do_initcall_level(level);
}
You can see the names above with their associated index: early is 0, core is 1, etc. Each of those __initcall*_start entries point to an array of function pointers which get called one after the other. Those function pointers are the actual modules and built-in initialization functions, the ones you specify with module_init, early_initcall, etc.
What determines which function pointer gets into which __initcall*_start array? The linker does this, using hints from the module_init and *_initcall macros. Those macros, for built-in modules, assign the function pointers to a specific ELF section.
Example with module_init
Considering a built-in module (configured with y in .config), module_init simply expands like this (include/linux/init.h):
#define module_init(x) __initcall(x);
and then we follow this:
#define __initcall(fn) device_initcall(fn)
#define device_initcall(fn) __define_initcall(fn, 6)
So, now, module_init(my_func) means __define_initcall(my_func, 6). This is _define_initcall:
#define __define_initcall(fn, id) \
static initcall_t __initcall_##fn##id __used \
__attribute__((__section__(".initcall" #id ".init"))) = fn
which means, so far, we have:
static initcall_t __initcall_my_func6 __used
__attribute__((__section__(".initcall6.init"))) = my_func;
Wow, lots of GCC stuff, but it only means that a new symbol is created, __initcall_my_func6, that's put in the ELF section named .initcall6.init, and as you can see, points to the specified function (my_func). Adding all the functions to this section eventually creates the complete array of function pointers, all stored within the .initcall6.init ELF section.
Initialization example
Look again at this chunk:
for (fn = initcall_levels[level]; fn < initcall_levels[level+1]; fn++)
do_one_initcall(*fn);
Let's take level 6, which represents all the built-in modules initialized with module_init. It starts from __initcall6_start, its value being the address of the first function pointer registered within the .initcall6.init section, and ends at __initcall7_start (excluded), incrementing each time with the size of *fn (which is an initcall_t, which is a void*, which is 32-bit or 64-bit depending on the architecture).
do_one_initcall will simply call the function pointed to by the current entry.
Within a specific initialization section, what determines why an initialization function is called before another is simply the order of the files within the Makefiles since the linker will concatenate the __initcall_* symbols one after the other in their respective ELF init. sections.
This fact is actually used in the kernel, e.g. with device drivers (drivers/Makefile):
# GPIO must come after pinctrl as gpios may need to mux pins etc
obj-y += pinctrl/
obj-y += gpio/
tl;dr: the Linux kernel initialization mechanism is really beautiful, albeit highlight GCC-dependent.
module_init is used to mark a function to be used as the entry-point of a Linux device-driver.
It is called
during do_initcalls() (for a builtin driver)
or
at module insertion time (for a *.ko module)
There can be ONLY 1 module_init() per driver module.
The *_initcall() functions are usually used to set the function-pointers for initialising various subsystems.
do_initcalls() within Linux kernel source code contains the invocation of the list of various initcalls and the relative order in which they are called during the Linux kernel boot-up.
early_initcall()
core_initcall()
postcore_initcall()
arch_initcall()
subsys_initcall()
fs_initcall()
device_initcall()
late_initcall()
end of built-in modules
modprobe or insmod of *.ko modules.
Using module_init() in a device driver is equivalent to registering a device_initcall().
Keep in mind that during compilation, the order of linking the various driver object files(*.o) within the Linux kernel is significant; it determines the order in which they are called at runtime.
*_initcall functions of the same level
will be called during boot in the order they are linked.
For example changing the link order of SCSI drivers in drivers/scsi/Makefile will change the order in which the SCSI controllers are detected, and thus the numbering of the disks.
It seems that no one has focused on how the linker script is configured to provide function pointers used for initialization to the kernel code, so let's try to look at how beautifully the Linux kernel creates linker script for init calls.
Because above great answers showed that how the Linux C code can create and manage all the initcalls in such as way that how to define a function as initcall, global variable to access the defined functions, and functions that actually invokes the defined initcall at the initialization phase, I don't want to revisit them again.
Therefore, here, we'd like to focus on how each element of the global array variable called initcall_levels[] is defined, what does it mean, what is contained in the memory pointed to by each element of the initcall_levels array, etc.
First, let's try to understand where the variables are defined in the Linux kernel repository. When you look at the init/main.c file, you can find that all elements of the initcall_levels array have not been defined in the main.c file and imported from somewhere.
extern initcall_t __initcall_start[];
extern initcall_t __initcall0_start[];
extern initcall_t __initcall1_start[];
extern initcall_t __initcall2_start[];
extern initcall_t __initcall3_start[];
extern initcall_t __initcall4_start[];
extern initcall_t __initcall5_start[];
extern initcall_t __initcall6_start[];
extern initcall_t __initcall7_start[];
extern initcall_t __initcall_end[];
However, you can find that those variables are not declared in any C source code of the Linux repository, then where the variables come from? From linker script!
Linux provides lots of helper functions to help programmers generate architecture-specific linker script file, and they are defined in linux/include/asm-generic/vmlinux.lds.h file which also provides helper for the initcalls.
#define __VMLINUX_SYMBOL(x) _##x
#define __VMLINUX_SYMBOL_STR(x) "_" #x
#else
#define __VMLINUX_SYMBOL(x) x
#define __VMLINUX_SYMBOL_STR(x) #x
#endif
/* Indirect, so macros are expanded before pasting. */
#define VMLINUX_SYMBOL(x) __VMLINUX_SYMBOL(x)
#define INIT_CALLS_LEVEL(level) \
VMLINUX_SYMBOL(__initcall##level##_start) = .; \
KEEP(*(.initcall##level##.init)) \
KEEP(*(.initcall##level##s.init)) \
#define INIT_CALLS \
VMLINUX_SYMBOL(__initcall_start) = .; \
KEEP(*(.initcallearly.init)) \
INIT_CALLS_LEVEL(0) \
INIT_CALLS_LEVEL(1) \
INIT_CALLS_LEVEL(2) \
INIT_CALLS_LEVEL(3) \
INIT_CALLS_LEVEL(4) \
INIT_CALLS_LEVEL(5) \
INIT_CALLS_LEVEL(rootfs) \
INIT_CALLS_LEVEL(6) \
INIT_CALLS_LEVEL(7) \
VMLINUX_SYMBOL(__initcall_end) = .;
We can easily find that several macros are defined for initcalls. Most important macro is INIT_CALLS that emits linker script syntax that defines a linker script symbol that can be accessed in the plain C code and input section.
In detail, each invocation of INIT_CALLS_LEVEL(x) macro defines a new symbol called __initcall##level_##start (refer ## concatenation operation in CPP); this symbol is generated by VMLINUX_SYMBOL(__initcall##level##_start) = .;. For example INIT_CALLS_LEVEL(1) macro defines linker script symbol named __initcall1_start.
As a result, symbols __initcall0_start to __initcall7_start are defined in the linker script and can be referenced in the C code by declaring it with the extern keyword.
Also, INIT_CALLS_LEVEL macro defines new sections called .initcallN.init, here N is a 0 to 7. The generated section contains all the functions defined with a provided macro such as __define_initcall as specified by the section attribute.
#define __define_initcall(fn, id) \
static initcall_t __initcall_##fn##id __used \
__attribute__((__section__(".initcall" #id ".init"))) = fn
The created symbols and sections should be configured correctly by the linker script to be located in one section, .init.data section. To enable this, the INIT_DATA_SECTION macro is used; and we can find that it invokes INIT_CALLS macro that we've looked.
#define INIT_DATA_SECTION(initsetup_align) \
.init.data : AT(ADDR(.init.data) - LOAD_OFFSET) { \
INIT_DATA \
INIT_SETUP(initsetup_align) \
INIT_CALLS \
CON_INITCALL \
SECURITY_INITCALL \
INIT_RAM_FS \
}
Therefore, by invoking INIT_CALLS macro, the Linux linker locates __initcall0_start to __initcall7_start symbols and the .initcall0.init to .initcall7.init sections in the .init.data section, which are located back to back. Here note that each symbol doesn't contain any data, but used to locate where the generated section starts and ends.
Then let's try to look at if the compiled Linux kernel correctly contains the generated symbols, sections, and the function. After compiling the Linux kernel, by making use of the nm tool, we can retrieve all symbols defined in the compiled Linux image called vmlinux.
//ordering nm result numerical order
$nm -n vmlinux > symbol
$vi symbol
ffffffff828ab1c8 T __initcall0_start
ffffffff828ab1c8 t __initcall_ipc_ns_init0
ffffffff828ab1d0 t __initcall_init_mmap_min_addr0
ffffffff828ab1d8 t __initcall_evm_display_config0
ffffffff828ab1e0 t __initcall_init_cpufreq_transition_notifier_list0
ffffffff828ab1e8 t __initcall_jit_init0
ffffffff828ab1f0 t __initcall_net_ns_init0
ffffffff828ab1f8 T __initcall1_start
ffffffff828ab1f8 t __initcall_xen_pvh_gnttab_setup1
ffffffff828ab200 t __initcall_e820__register_nvs_regions1
ffffffff828ab208 t __initcall_cpufreq_register_tsc_scaling1
......
ffffffff828ab3a8 t __initcall___gnttab_init1s
ffffffff828ab3b0 T __initcall2_start
ffffffff828ab3b0 t __initcall_irq_sysfs_init2
ffffffff828ab3b8 t __initcall_audit_init2
ffffffff828ab3c0 t __initcall_bdi_class_init2
As shown in the above, in between __initcall0_start and __initcall2_start symbol, all functions defined with pure_initcall macro are located. For example, let's look at the ipc_ns_init function defined in ipc/shim.c file
static int __init ipc_ns_init(void)
{
const int err = shm_init_ns(&init_ipc_ns);
WARN(err, "ipc: sysv shm_init_ns failed: %d\n", err);
return err;
}
pure_initcall(ipc_ns_init);
As shown in the above, pure_initcall macro is used to put ipc_ns_init function in the .initcall0.init section which is located by the __initcall0_start symbol. Therefore, as shown in the below code, all the functions in the .initcallN.init sections are invoked one by one sequentially.
for (fn = initcall_levels[level]; fn < initcall_levels[level+1]; fn++)
do_one_initcall(*fn);
I would like to know the meaning of %P3 in the following code:
#define get_user(x, ptr) \
({ \
int __ret_gu; \
register __inttype(*(ptr)) __val_gu asm("%edx"); \
__chk_user_ptr(ptr); \
might_fault(); \
asm volatile("call __get_user_%P3" \
: "=a" (__ret_gu), "=r" (__val_gu) \
: "0" (ptr), "i" (sizeof(*(ptr)))); \
(x) = (__typeof__(*(ptr))) __val_gu; \
__ret_gu; \
})
Moreover in the LLVM IR the code is mapped to:
call { i32*, i64 } asm sideeffect "call __get_user_${3:P}", "={ax},={edx},0,i,~{dirflag},~{fpsr},~{flags}"(i32* %tmp73, i64 4)
My understanding is that this is actually calling a specific function __get_user_X in arch/x86/lib/getuser.S, but it's not clear which one in particular (__get_user_4?).
Finally, I would like to understand the difference between %P and %p.
I think %P3 indicates that the X in __get_user_X is depend on "i" (sizeof((ptr)). Such as sizeof((ptr)) may be 1, 2, 4, 8.
The 3 means the third parameter inside the asm volatile("...") statement. And %P is for string concatenation.
About %P and %p difference, I guess it's for string specification, but I am not sure. I copied following sentences from GCC user manual:
%p Substitutes the standard macro predefinitions for the current target machine.
Use this when running cpp.
%P Like ‘%p’, but puts ‘__’ before and after the name of each predefined macro,
except for macros that start with ‘__’ or with ‘_L’, where L is an uppercase
letter. This is for ISO C.