Unraid-SlackPack/source/freeipmi/usr/share/doc/freeipmi/freeipmi-coding.txt
2016-03-24 14:02:30 -06:00

273 lines
12 KiB
Plaintext

FreeIPMI Coding
by
Albert Chu
chu11@llnl.gov
Last Updated: February 8, 2012
These are some short descriptions on coding style, API style, other
thoughts for those interested in developing for FreeIPMI.
1) Code Style
-------------
The GNU coding style was selected for FreeIPMI. Please try to follow
the coding style used in the rest of FreeIPMI. Here's a short example
that covers the generics of the GNU coding style.
int
main(int argc, char **argv)
{
int a = 0;
int b = 1;
if (a == 1)
printf("yoda\n");
if (a == 5
|| b == 1)
{
printf("foobar\n");
printf("xyzzy\n");
}
while (a++ < 5)
printf("lala\n");
while (b++ < 7)
{
printf("blah\n");
printf("garble\n");
}
}
2) Parameter Checking
---------------------
Please carefully check the input parameters on the inputs your
program and/or functions take. Minor parsing issues can lead to
catastrophic mistakes in IPMI.
For example, suppose you have a --power-control option that takes a
number to represent a type of operation (on, off, etc.). Suppose a
user inputs "--power-control=foobar". The "foobar" will be read as a
'0' by strtoul(). If not properly checked, the '0' can be passed to
the IPMI Chassis Control command, which uses the '0' to power off a
node.
In programs, when appropriate, output error messages to the user
indicating that how and why the inputted parameters were incorrect.
3) Code Consistency
-------------------
Please keep your code as consistent as possible to other code in
FreeIPMI. That includes code indenting style, brace style, API style,
and naming convention (which is discussed in more detail below).
Although there may be situations that a particular API style or naming
convention will make things easier for you and your code (such as
shortening the name of a function, decreasing the number of parameters
you need to pass to a function via a struct, etc.), we ask that your
code be consistent so that it does not confuse other developers.
If there is a distinct technical reason that you must use a different
API style, please bring it up with the FreeIPMI authors.
For example, pretty much all of the "fill" functions in libfreeipmi
take the exact parameters they need to fill the fiid object which is
passed along as a parameter. All parameters are passed by value, not
by a pointer or other method (e.g. object, struct, etc.).
Exceptions do exist. For example, fill_cmd_chassis_identify() takes
parameters passed by pointer instead of passed by value. The reason
is that both fields are optional and need not be filled according to
the IPMI specification. The pointer gives the caller the ability to
set values (by passing a valid pointer) or not (by passing NULL).
4) Libfreeipmi naming/function parameter conventions
----------------------------------------------------
The naming style in libfreeipmi was developed primarily for the
purpose of readability when code is being compared to the IPMI
specification.
Due to the size of the IPMI spec, there will be a lot of code. In
earlier versions of FreeIPMI, there was confusion on where code was
located, what parameters were called, how parameters should be input,
etc. due to different people using different abbreviations styles,
putting functions out of order with the spec, in different files,
using/not-using different bitmasks, etc. The code has been auditted
and cleaned up since then.
So when adding new functions/templates/parameters/files/etc. to
libfreeipmi, please name them consistently to the rest of the
libfreeipmi library and the IPMI specification.
This includes:
- naming functions/templates/parameters/files based on the spec
- in most cases, not abbreviating any words (or using consistent
abbreviations in the rest of the library, check first!)
- matching parameter lists to the templates and in the same order
- ordering functions/templates/parameters/files/etc. consistently with the spec.
For example:
ipmi-messaging-support-cmds.c
is the file for messaging support commands, chapter 22 of the IPMI 2.0 spec.
tmpl_cmd_get_channel_authentication_capabilities_rq
tmpl_cmd_get_channel_authentication_capabilities_rs
are the templates for the Get Channel Authentication Capapilities
command.
fiid_template_t tmpl_cmd_get_channel_authentication_capabilities_rq =
{
{ 8, "cmd", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 4, "channel_number", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 3, "reserved1", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "get_ipmi_v2.0_extended_data", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 4, "maximum_privilege_level", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 4, "reserved2", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 0, "", 0}
};
fiid_template_t tmpl_cmd_get_channel_authentication_capabilities_rs =
{
{ 8, "cmd", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED | FIID_FIELD_MAKES_PACKET_SUFFICIENT},
{ 8, "comp_code", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED | FIID_FIELD_MAKES_PACKET_SUFFICIENT},
{ 8, "channel_number", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.none", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.md2", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.md5", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.reserved1", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.straight_password_key", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.oem_prop", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.reserved2", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_type.ipmi_v2.0_extended_capabilities_available", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_status.anonymous_login", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_status.null_username", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_status.non_null_username", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_status.user_level_authentication", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_status.per_message_authentication", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "authentication_status.k_g", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 2, "authentication_status.reserved", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "channel_supports_ipmi_v1.5_connections", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 1, "channel_supports_ipmi_v2.0_connections", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 6, "reserved", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 24, "oem_id", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 8, "oem_auxiliary_data", FIID_FIELD_REQUIRED | FIID_FIELD_LENGTH_FIXED},
{ 0, "", 0}
};
The fields listed in the templates above directly match table 22-15,
the Get Channel Authentication Capabilities request and response
commands.
int fill_cmd_get_channel_authentication_capabilities (uint8_t channel_number,
uint8_t maximum_privilege_level,
fiid_obj_t obj_cmd_rq);
The function above matches the naming and takes exactly the parameters
needed by the Get Channel Authentication Capabilities request
template.
The coding conditions specified above may lead to function names or
function parameters names that exceed the 80 column mark or having
very long parameter lists. We accept this annoyance (or poor coding
style, we admit it), as we consider matching the specification as a
more important need in libfreeipmi.
For example:
int
fill_cmd_set_lan_configuration_parameters_authentication_type_enables (uint8_t channel_number,
uint8_t callback_level_none,
uint8_t callback_level_md2,
uint8_t callback_level_md5,
uint8_t callback_level_straight_password,
uint8_t callback_level_oem_proprietary,
uint8_t user_level_none,
uint8_t user_level_md2,
uint8_t user_level_md5,
uint8_t user_level_straight_password,
uint8_t user_level_oem_proprietary,
uint8_t operator_level_none,
uint8_t operator_level_md2,
uint8_t operator_level_md5,
uint8_t operator_level_straight_password,
uint8_t operator_level_oem_proprietary,
uint8_t admin_level_none,
uint8_t admin_level_md2,
uint8_t admin_level_md5,
uint8_t admin_level_straight_password,
uint8_t admin_level_oem_proprietary,
uint8_t oem_level_none,
uint8_t oem_level_md2,
uint8_t oem_level_md5,
uint8_t oem_level_straight_password,
uint8_t oem_level_oem_proprietary,
fiid_obj_t obj_cmd_rq);
The function name and parameters look pretty long and terrible. But
the names and fields exactly match the get authentication type enables
fields listed in Table 23-4. There should be very little difficulty
understanding what this funciton does, how it should be called, and
what the parameters are if you are reading along with the spec.
Because we want the code to match the IPMI spec as closely as
possible, we currently accept the code inefficiencies (due to large
stacks of parameters) that come with having long parameters lists and
the atrocities of having gigantic 25+ parameter function calls in
code.
5) Compilation and Development Notes
------------------------------------
Normally, to build and develop from the source tree you need to run:
./autogen.sh
./configure
make
out of the directory you are in.
The following configure options can help you with many development
issues:
--enable-debug
This option will turn on compilation of debugging symbols. More
warnings may also be enabled when compiling.
In addition, this option may enable a few "workarounds" to allow for
easier development of code. Most notably, inband drivers may look in
additional locations for an ipckey, so that inband development can be
done in a local directory without a full installation.
Generally speaking, if developing out of a local tree, you'll almost
always want to enable this flag.
--enable-profile
If --enable-debug is enabled, this option will turn on compilation
with profiling data.
--enable-trace
Compile tracing information into a significant portion of FreeIPMI
code. This will result in significant amounts of extra information to
be output.
--enable-rawdumps
Some users may be more familiar with raw IPMI packet dumps instead of
the detailed field identified dumps that normally occur when running
tools with --debug. In this case, --enable-rawdumps, will inform
FreeIPMI to also output packets in raw form.