| 
									
										
										
										
											2005-06-02 19:38:55 +00:00
										 |  |  | == Asterisk Patch/Coding Guidelines == | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | -------------------------------------- | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | We are looking forward to your contributions to Asterisk - the  | 
					
						
							|  |  |  | Open Source PBX! As Asterisk is a large and in some parts very | 
					
						
							|  |  |  | time-sensitive application, the code base needs to conform to | 
					
						
							|  |  |  | a common set of coding rules so that many developers can enhance | 
					
						
							|  |  |  | and maintain the code. Code also needs to be reviewed and tested | 
					
						
							|  |  |  | so that it works and follows the general architecture and guide- | 
					
						
							|  |  |  | lines, and is well documented. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Asterisk is published under a dual-licensing scheme by Digium. | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | To be accepted into the codebase, all non-trivial changes must be | 
					
						
							|  |  |  | disclaimed to Digium or placed in the public domain. For more information | 
					
						
							|  |  |  | see http://bugs.digium.com | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-01 19:34:50 +00:00
										 |  |  | Patches should be in the form of a unified (-u) diff, made from the directory | 
					
						
							|  |  |  | above the top-level Asterisk source directory. For example: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - the base code you are working from is in ~/work/asterisk-base | 
					
						
							|  |  |  | - the changes are in ~/work/asterisk-new | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~/work$ diff -urN asterisk-base asterisk-new | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | If you are changing within a fresh CVS/SVN repository, you can create | 
					
						
							|  |  |  | a patch with | 
					
						
							|  |  |  | $ cvs diff -urN <mycodefile>.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * General rules | 
					
						
							|  |  |  | --------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - All code, filenames, function names and comments must be in ENGLISH. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Don't annotate your changes with comments like "/* JMG 4/20/04 */"; | 
					
						
							|  |  |  |   Comments should explain what the code does, not when something was changed | 
					
						
							|  |  |  |   or who changed it. If you have done a larger contribution, make sure | 
					
						
							|  |  |  |   that you are added to the CREDITS file. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Don't make unnecessary whitespace changes throughout the code. | 
					
						
							|  |  |  |   If you make changes, submit them to the tracker as separate patches | 
					
						
							|  |  |  |   that only include whitespace and formatting changes. | 
					
						
							| 
									
										
										
										
											2004-05-15 15:34:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Don't use C++ type (//) comments. | 
					
						
							| 
									
										
										
										
											2004-05-15 15:34:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Try to match the existing formatting of the file you are working on. | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Use spaces instead of tabs when aligning in-line comments or #defines (this makes  | 
					
						
							|  |  |  |   your comments aligned even if the code is viewed with another tabsize) | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | * Declaration of functions and variables | 
					
						
							|  |  |  | ---------------------------------------- | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Do not declare variables mid-block (e.g. like recent GNU compilers support) | 
					
						
							|  |  |  |   since it is harder to read and not portable to GCC 2.95 and others. | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Functions and variables that are not intended to be used outside the module | 
					
						
							|  |  |  |   must be declared static. | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - When reading integer numeric input with scanf (or variants), do _NOT_ use '%i' | 
					
						
							|  |  |  |   unless you specifically want to allow non-base-10 input; '%d' is always a better | 
					
						
							|  |  |  |   choice, since it will not silently turn numbers with leading zeros into base-8. | 
					
						
							| 
									
										
										
										
											2005-04-23 15:11:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Strings that are coming from input should not be used as a first argument to | 
					
						
							|  |  |  |   a formatted *printf function. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * Use the internal API | 
					
						
							|  |  |  | ---------------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Make sure you are aware of the string and data handling functions that exist | 
					
						
							|  |  |  |   within Asterisk to enhance portability and in some cases to produce more | 
					
						
							|  |  |  |   secure and thread-safe code. Check utils.c/utils.h for these. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * Code formatting | 
					
						
							|  |  |  | ----------------- | 
					
						
							| 
									
										
										
										
											2005-07-19 17:59:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-19 04:31:02 +00:00
										 |  |  | Roughly, Asterisk code formatting guidelines are generally equivalent to the  | 
					
						
							| 
									
										
										
										
											2004-11-08 02:49:33 +00:00
										 |  |  | following: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-12 04:28:58 +00:00
										 |  |  | # indent -i4 -ts4 -br -brs -cdw -lp -ce -nbfda -npcs -nprs -npsl -nbbo -saf -sai -saw -cs -ln90 foo.c | 
					
						
							| 
									
										
										
										
											2005-07-19 17:59:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | this means in verbose: | 
					
						
							|  |  |  |  -i4:    indent level 4 | 
					
						
							|  |  |  |  -ts4:   tab size 4 | 
					
						
							|  |  |  |  -br:    braces on if line | 
					
						
							|  |  |  |  -brs:   braces on struct decl line | 
					
						
							|  |  |  |  -cdw:   cuddle do while | 
					
						
							| 
									
										
										
										
											2006-02-12 04:28:58 +00:00
										 |  |  |  -lp:    line up continuation below parenthesis | 
					
						
							| 
									
										
										
										
											2005-07-19 17:59:27 +00:00
										 |  |  |  -ce:    cuddle else | 
					
						
							|  |  |  |  -nbfda: dont break function decl args | 
					
						
							|  |  |  |  -npcs:  no space after function call names | 
					
						
							|  |  |  |  -nprs:  no space after parentheses | 
					
						
							|  |  |  |  -npsl:  dont break procedure type | 
					
						
							|  |  |  |  -saf:   space after for | 
					
						
							|  |  |  |  -sai:   space after if | 
					
						
							|  |  |  |  -saw:   space after while | 
					
						
							| 
									
										
										
										
											2006-02-12 04:28:58 +00:00
										 |  |  |  -cs:    space after cast | 
					
						
							|  |  |  |  -ln90:  line length 90 columns | 
					
						
							| 
									
										
										
										
											2004-11-08 02:49:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | Function calls and arguments should be spaced in a consistent way across | 
					
						
							|  |  |  | the codebase. | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | 	GOOD: foo(arg1, arg2); | 
					
						
							|  |  |  | 	GOOD: foo(arg1,arg2);	/* Acceptable but not preferred */ | 
					
						
							|  |  |  | 	BAD: foo (arg1, arg2); | 
					
						
							|  |  |  | 	BAD: foo( arg1, arg2 ); | 
					
						
							|  |  |  | 	BAD: foo(arg1, arg2,arg3); | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Don't treat keywords (if, while, do, return) as if they were functions; | 
					
						
							|  |  |  | leave space between the keyword and the expression used (if any). For 'return', | 
					
						
							|  |  |  | don't even put parentheses around the expression, since they are not | 
					
						
							|  |  |  | required. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | There is no shortage of whitespace characters :-) Use them when they make | 
					
						
							|  |  |  | the code easier to read. For example: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | 	for (str=foo;str;str=str->next) | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | is harder to read than | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | 	for (str = foo; str; str = str->next) | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | Following are examples of how code should be formatted. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Functions: | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | int foo(int a, char *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - If statements: | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | if (foo) { | 
					
						
							|  |  |  | 	bar(); | 
					
						
							|  |  |  | } else { | 
					
						
							|  |  |  | 	blah(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Case statements: | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | switch (foo) { | 
					
						
							| 
									
										
										
										
											2004-05-18 05:52:52 +00:00
										 |  |  | case BAR: | 
					
						
							|  |  |  | 	blah(); | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | case OTHER: | 
					
						
							|  |  |  | 	other(); | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - No nested statements without braces, e.g.: | 
					
						
							| 
									
										
										
										
											2004-05-18 05:52:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | for (x = 0; x < 5; x++) | 
					
						
							| 
									
										
										
										
											2004-05-18 05:52:52 +00:00
										 |  |  | 	if (foo)  | 
					
						
							|  |  |  | 		if (bar) | 
					
						
							|  |  |  | 			baz(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | instead do: | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | for (x = 0; x < 5; x++) { | 
					
						
							| 
									
										
										
										
											2004-05-18 05:52:52 +00:00
										 |  |  | 	if (foo) { | 
					
						
							|  |  |  | 		if (bar) | 
					
						
							|  |  |  | 			baz(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-05-02 19:24:05 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Don't build code like this: | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-19 04:31:02 +00:00
										 |  |  | if (foo) { | 
					
						
							| 
									
										
										
										
											2005-06-07 21:28:56 +00:00
										 |  |  | 	/* .... 50 lines of code ... */ | 
					
						
							| 
									
										
										
										
											2005-05-19 04:31:02 +00:00
										 |  |  | } else { | 
					
						
							| 
									
										
										
										
											2005-06-07 21:28:56 +00:00
										 |  |  | 	result = 0; | 
					
						
							|  |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2005-05-19 04:31:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Instead, try to minimize the number of lines of code that need to be | 
					
						
							|  |  |  | indented, by only indenting the shortest case of the 'if' | 
					
						
							|  |  |  | statement, like so: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if !(foo) { | 
					
						
							| 
									
										
										
										
											2005-06-07 21:28:56 +00:00
										 |  |  | 	result = 0; | 
					
						
							|  |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2005-05-19 04:31:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | .... 50 lines of code .... | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | When this technique is used properly, it makes functions much easier to read | 
					
						
							|  |  |  | and follow, especially those with more than one or two 'setup' operations | 
					
						
							|  |  |  | that must succeed for the rest of the function to be able to execute. | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Labels/goto are acceptable | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Proper use of this technique may occasionally result in the need for a | 
					
						
							|  |  |  | label/goto combination so that error/failure conditions can exit the | 
					
						
							|  |  |  | function while still performing proper cleanup. This is not a bad thing! | 
					
						
							|  |  |  | Use of goto in this situation is encouraged, since it removes the need | 
					
						
							|  |  |  | for excess code indenting without requiring duplication of cleanup code. | 
					
						
							| 
									
										
										
										
											2005-05-19 04:31:02 +00:00
										 |  |  |         | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Never use an uninitialized variable | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | Make sure you never use an uninitialized variable.  The compiler will  | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | usually warn you if you do so. However, do not go too far the other way, | 
					
						
							|  |  |  | and needlessly initialize variables that do not require it. If the first | 
					
						
							|  |  |  | time you use a variable in a function is to store a value there, then | 
					
						
							|  |  |  | initializing it at declaration is pointless, and will generate extra | 
					
						
							|  |  |  | object code and data in the resulting binary with no purpose. When in doubt, | 
					
						
							|  |  |  | trust the compiler to tell you when you need to initialize a variable; | 
					
						
							|  |  |  | if it does not warn you, initialization is not needed. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Do not cast 'void *' | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Do not explicitly cast 'void *' into any other type, nor should you cast any | 
					
						
							|  |  |  | other type into 'void *'. Implicit casts to/from 'void *' are explicitly | 
					
						
							|  |  |  | allowed by the C specification. This means the results of malloc(), calloc(), | 
					
						
							|  |  |  | alloca(), and similar functions do not _ever_ need to be cast to a specific | 
					
						
							|  |  |  | type, and when you are passing a pointer to (for example) a callback function | 
					
						
							|  |  |  | that accepts a 'void *' you do not need to cast into that type. | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | * Variable naming | 
					
						
							|  |  |  | ----------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Global variables | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | Name global variables (or local variables when you have a lot of them or | 
					
						
							|  |  |  | are in a long function) something that will make sense to aliens who | 
					
						
							|  |  |  | find your code in 100 years.  All variable names should be in lower  | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | case, except when following external APIs or specifications that normally | 
					
						
							|  |  |  | use upper- or mixed-case variable names; in that situation, it is | 
					
						
							|  |  |  | preferable to follow the external API/specification for ease of | 
					
						
							|  |  |  | understanding. | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | Make some indication in the name of global variables which represent | 
					
						
							|  |  |  | options that they are in fact intended to be global. | 
					
						
							|  |  |  |  e.g.: static char global_something[80] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Don't use un-necessary typedef's | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Don't use 'typedef' just to shorten the amount of typing; there is no substantial | 
					
						
							|  |  |  | benefit in this: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct foo { | 
					
						
							|  |  |  | 	int bar; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | typedef foo_t struct foo; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | In fact, don't use 'variable type' suffixes at all; it's much preferable to | 
					
						
							|  |  |  | just type 'struct foo' rather than 'foo_s'. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Use enums instead of #define where possible | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Use enums rather than long lists of #define-d numeric constants when possible; | 
					
						
							|  |  |  | this allows structure members, local variables and function arguments to | 
					
						
							|  |  |  | be declared as using the enum's type. For example: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum option { | 
					
						
							|  |  |  |   OPT_FOO = 1 | 
					
						
							|  |  |  |   OPT_BAR = 2 | 
					
						
							|  |  |  |   OPT_BAZ = 4 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static enum option global_option; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static handle_option(const enum option opt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ... | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Note: The compiler will _not_ force you to pass an entry from the enum | 
					
						
							|  |  |  | as an argument to this function; this recommendation serves only to make | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | the code clearer and somewhat self-documenting. In addition, when using | 
					
						
							|  |  |  | switch/case blocks that switch on enum values, the compiler will warn | 
					
						
							|  |  |  | you if you forget to handle one or more of the enum values, which can be | 
					
						
							|  |  |  | handy. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * String handling | 
					
						
							|  |  |  | ----------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Don't use strncpy for copying whole strings; it does not guarantee that the | 
					
						
							|  |  |  | output buffer will be null-terminated. Use ast_copy_string instead, which | 
					
						
							|  |  |  | is also slightly more efficient (and allows passing the actual buffer | 
					
						
							|  |  |  | size, which makes the code clearer). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Don't use ast_copy_string (or any length-limited copy function) for copying | 
					
						
							|  |  |  | fixed (known at compile time) strings into buffers, if the buffer is something | 
					
						
							|  |  |  | that has been allocated in the function doing the copying. In that case, you | 
					
						
							|  |  |  | know at the time you are writing the code whether the buffer is large enough | 
					
						
							|  |  |  | for the fixed string or not, and if it's not, your code won't work anyway! | 
					
						
							|  |  |  | Use strcpy() for this operation, or directly set the first two characters | 
					
						
							|  |  |  | of the buffer if you are just trying to store a one-character string in the | 
					
						
							|  |  |  | buffer. If you are trying to 'empty' the buffer, just store a single | 
					
						
							|  |  |  | NULL character ('\0') in the first byte of the buffer; nothing else is | 
					
						
							|  |  |  | needed, and any other method is wasteful. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | In addition, if the previous operations in the function have already | 
					
						
							|  |  |  | determined that the buffer in use is adequately sized to hold the string | 
					
						
							|  |  |  | you wish to put into it (even if you did not allocate the buffer yourself), | 
					
						
							|  |  |  | use a direct strcpy(), as it can be inlined and optimized to simple | 
					
						
							|  |  |  | processor operations, unlike ast_copy_string(). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | * Use of functions | 
					
						
							|  |  |  | ------------------ | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | When making applications, always ast_strdupa(data) to a local pointer if | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | you intend to parse the incoming data string. | 
					
						
							| 
									
										
										
										
											2005-06-07 21:28:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (data) | 
					
						
							|  |  |  | 		mydata = ast_strdupa(data); | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | - Separating arguments to dialplan applications and functions | 
					
						
							| 
									
										
										
										
											2005-11-10 23:08:00 +00:00
										 |  |  | Use ast_app_separate_args() to separate the arguments to your application | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | once you have made a local copy of the string. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Parsing strings with strsep | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Use strsep() for parsing strings when possible; there is no worry about | 
					
						
							|  |  |  | 're-entrancy' as with strtok(), and even though it modifies the original | 
					
						
							|  |  |  | string (which the man page warns about), in many cases that is exactly | 
					
						
							|  |  |  | what you want! | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Create generic code! | 
					
						
							|  |  |  | If you do the same or a similar operation more than one time, make it a | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | function or macro. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Make sure you are not duplicating any functionality already found in an | 
					
						
							|  |  |  | API call somewhere.  If you are duplicating functionality found in  | 
					
						
							|  |  |  | another static function, consider the value of creating a new API call  | 
					
						
							|  |  |  | which can be shared. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | * Handling of pointers and allocations | 
					
						
							|  |  |  | -------------------------------------- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Dereference or localize pointers | 
					
						
							|  |  |  | Always dereference or localize pointers to things that are not yours like | 
					
						
							|  |  |  | channel members in a channel that is not associated with the current  | 
					
						
							|  |  |  | thread and for which you do not have a lock. | 
					
						
							|  |  |  | 	channame = ast_strdupa(otherchan->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Use const on pointer arguments if possible | 
					
						
							|  |  |  | Use const on pointer arguments which your function will not be modifying, as this  | 
					
						
							|  |  |  | allows the compiler to make certain optimizations. In general, use 'const' | 
					
						
							|  |  |  | on any argument that you have no direct intention of modifying, as it can | 
					
						
							|  |  |  | catch logic/typing errors in your code when you use the argument variable | 
					
						
							|  |  |  | in a way that you did not intend. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Do not create your own linked list code - reuse! | 
					
						
							| 
									
										
										
										
											2005-07-10 23:58:07 +00:00
										 |  |  | As a common example of this point, make an effort to use the lockable | 
					
						
							|  |  |  | linked-list macros found in include/asterisk/linkedlists.h. They are | 
					
						
							|  |  |  | efficient, easy to use and provide every operation that should be | 
					
						
							|  |  |  | necessary for managing a singly-linked list (if something is missing, | 
					
						
							|  |  |  | let us know!). Just because you see other open-coded list implementations | 
					
						
							|  |  |  | in the source tree is no reason to continue making new copies of | 
					
						
							|  |  |  | that code... There are also a number of common string manipulation | 
					
						
							|  |  |  | and timeval manipulation functions in asterisk/strings.h and asterisk/time.h; | 
					
						
							|  |  |  | use them when possible. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | - Avoid needless allocations! | 
					
						
							| 
									
										
										
										
											2005-07-10 23:51:10 +00:00
										 |  |  | Avoid needless malloc(), strdup() calls. If you only need the value in | 
					
						
							|  |  |  | the scope of your function try ast_strdupa() or declare structs on the | 
					
						
							|  |  |  | stack and pass a pointer to them. However, be careful to _never_ call | 
					
						
							|  |  |  | alloca(), ast_strdupa() or similar functions in the argument list | 
					
						
							|  |  |  | of a function you are calling; this can cause very strange stack | 
					
						
							|  |  |  | arrangements and produce unexpected behavior. | 
					
						
							| 
									
										
										
										
											2005-01-18 23:35:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | -Allocations for structures | 
					
						
							| 
									
										
										
										
											2006-01-10 23:51:42 +00:00
										 |  |  | When allocating/zeroing memory for a structure, use code like this: | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct foo *tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ... | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-10 23:51:42 +00:00
										 |  |  | tmp = ast_calloc(1, sizeof(*tmp)); | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-10 23:51:42 +00:00
										 |  |  | Avoid the combination of ast_malloc() and memset().  Instead, always use | 
					
						
							|  |  |  | ast_calloc(). This will allocate and zero the memory in a single operation.  | 
					
						
							|  |  |  | In the case that uninitialized memory is acceptable, there should be a comment | 
					
						
							|  |  |  | in the code that states why this is the case. | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-10 23:51:42 +00:00
										 |  |  | Using sizeof(*tmp) instead of sizeof(struct foo) eliminates duplication of the  | 
					
						
							|  |  |  | 'struct foo' identifier, which makes the code easier to read and also ensures  | 
					
						
							|  |  |  | that if it is copy-and-pasted it won't require as much editing. | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-10 23:51:42 +00:00
										 |  |  | The ast_* family of functions for memory allocation are functionally the same. | 
					
						
							|  |  |  | They just add an Asterisk log error message in the case that the allocation | 
					
						
							|  |  |  | fails for some reason. This eliminates the need to generate custom messages | 
					
						
							|  |  |  | throughout the code to log that this has occurred. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -String Duplications | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The functions strdup and strndup can *not* accept a NULL argument. This results | 
					
						
							|  |  |  | in having code like this: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (str) | 
					
						
							|  |  |  | 		newstr = strdup(str); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		newstr = NULL; | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-10 23:51:42 +00:00
										 |  |  | However, the ast_strdup and ast_strdup functions will happily accept a NULL | 
					
						
							|  |  |  | argument without generating an error.  The same code can be written as: | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2006-01-11 00:06:15 +00:00
										 |  |  | 	newstr = ast_strdup(str); | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-13 18:38:55 +00:00
										 |  |  | Furthermore, it is unnecessary to have code that malloc/calloc's for the length | 
					
						
							|  |  |  | of a string (+1 for the terminating '\0') and then using strncpy to copy the | 
					
						
							|  |  |  | copy the string into the resulting buffer.  This is the exact same thing as | 
					
						
							|  |  |  | using ast_strdup. | 
					
						
							| 
									
										
										
										
											2005-06-02 19:17:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | * CLI Commands | 
					
						
							|  |  |  | -------------- | 
					
						
							| 
									
										
										
										
											2005-04-01 04:38:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | New CLI commands should be named using the module's name, followed by a verb | 
					
						
							|  |  |  | and then any parameters that the command needs. For example: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | *CLI> iax2 show peer <peername> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | not | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | *CLI> show iax2 peer <peername> | 
					
						
							| 
									
										
										
										
											2005-05-01 19:34:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | * New dialplan applications/functions | 
					
						
							|  |  |  | ------------------------------------- | 
					
						
							| 
									
										
										
										
											2005-05-01 19:34:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | There are two methods of adding functionality to the Asterisk | 
					
						
							|  |  |  | dialplan: applications and functions. Applications (found generally in | 
					
						
							|  |  |  | the apps/ directory) should be collections of code that interact with | 
					
						
							|  |  |  | a channel and/or user in some significant way. Functions (which can be | 
					
						
							|  |  |  | provided by any type of module) are used when the provided | 
					
						
							|  |  |  | functionality is simple... getting/retrieving a value, for | 
					
						
							|  |  |  | example. Functions should also be used when the operation is in no way | 
					
						
							|  |  |  | related to a channel (a computation or string operation, for example). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Applications are registered and invoked using the | 
					
						
							|  |  |  | ast_register_application function; see the apps/app_skel.c file for an | 
					
						
							|  |  |  | example. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-15 23:26:45 +00:00
										 |  |  | Functions are registered using 'struct ast_custom_function' | 
					
						
							| 
									
										
										
										
											2005-05-01 19:34:50 +00:00
										 |  |  | structures and the ast_custom_function_register function. | 
					
						
							| 
									
										
										
										
											2005-06-02 19:38:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | * Doxygen API Documentation Guidelines | 
					
						
							|  |  |  | -------------------------------------- | 
					
						
							| 
									
										
										
										
											2005-06-02 19:38:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | When writing Asterisk API documentation the following format should be | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | followed. Do not use the javadoc style. | 
					
						
							| 
									
										
										
										
											2005-06-02 19:38:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*! | 
					
						
							|  |  |  |  * \brief Do interesting stuff. | 
					
						
							|  |  |  |  * \param thing1 interesting parameter 1. | 
					
						
							|  |  |  |  * \param thing2 interesting parameter 2. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function does some interesting stuff. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \return zero on success, -1 on error. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int ast_interesting_stuff(int thing1, int thing2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Notice the use of the \param, \brief, and \return constructs.  These should be | 
					
						
							|  |  |  | used to describe the corresponding pieces of the function being documented. | 
					
						
							|  |  |  | Also notice the blank line after the last \param directive.  All doxygen | 
					
						
							|  |  |  | comments must be in one /*! */ block.  If the function or struct does not need | 
					
						
							|  |  |  | an extended description it can be left out. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Please make sure to review the doxygen manual and make liberal use of the \a, | 
					
						
							|  |  |  | \code, \c, \b, \note, \li and \e modifiers as appropriate. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | When documenting a 'static' function or an internal structure in a module, | 
					
						
							|  |  |  | use the \internal modifier to ensure that the resulting documentation | 
					
						
							|  |  |  | explicitly says 'for internal use only'. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Structures should be documented as follows. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! | 
					
						
							|  |  |  |  * \brief A very interesting structure. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct interesting_struct | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/*! \brief A data member. */ | 
					
						
							|  |  |  | 	int member1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int member2; /*!< \brief Another data member. */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Note that /*! */ blocks document the construct immediately following them | 
					
						
							|  |  |  | unless they are written, /*!< */, in which case they document the construct | 
					
						
							|  |  |  | preceding them. | 
					
						
							| 
									
										
										
										
											2005-10-31 22:47:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | * Finishing up before you submit your code | 
					
						
							|  |  |  | ------------------------------------------ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Look at the code once more | 
					
						
							|  |  |  | When you achieve your desired functionalty, make another few refactor | 
					
						
							|  |  |  | passes over the code to optimize it. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Read the patch | 
					
						
							|  |  |  | Before submitting a patch, *read* the actual patch file to be sure that  | 
					
						
							|  |  |  | all the changes you expect to be there are, and that there are no  | 
					
						
							|  |  |  | surprising changes you did not expect. During your development, that | 
					
						
							|  |  |  | part of Asterisk may have changed, so make sure you compare with the | 
					
						
							|  |  |  | latest CVS. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Listen to advice | 
					
						
							|  |  |  | If you are asked to make changes to your patch, there is a good chance | 
					
						
							|  |  |  | the changes will introduce bugs, check it even more at this stage. | 
					
						
							|  |  |  | Also remember that the bug marshal or co-developer that adds comments  | 
					
						
							|  |  |  | is only human, they may be in error :-) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - Optimize, optimize, optimize | 
					
						
							|  |  |  | If you are going to reuse a computed value, save it in a variable | 
					
						
							|  |  |  | instead of recomputing it over and over.  This can prevent you from  | 
					
						
							|  |  |  | making a mistake in subsequent computations, making it easier to correct | 
					
						
							|  |  |  | if the formula has an error and may or may not help optimization but  | 
					
						
							|  |  |  | will at least help readability. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Just an example (so don't over analyze it, that'd be a shame): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *prefix = "pre";	 | 
					
						
							|  |  |  | const char *postfix = "post"; | 
					
						
							|  |  |  | char *newname; | 
					
						
							|  |  |  | char *name = "data"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (name && (newname = alloca(strlen(name) + strlen(prefix) + strlen(postfix) + 3))) | 
					
						
							|  |  |  | 	snprintf(newname, strlen(name) + strlen(prefix) + strlen(postfix) + 3, "%s/%s/%s", prefix, name, postfix); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ...vs this alternative: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char *prefix = "pre"; | 
					
						
							|  |  |  | const char *postfix = "post"; | 
					
						
							|  |  |  | char *newname; | 
					
						
							|  |  |  | char *name = "data"; | 
					
						
							|  |  |  | int len = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (name && (len = strlen(name) + strlen(prefix) + strlen(postfix) + 3) && (newname = alloca(len))) | 
					
						
							|  |  |  | 	snprintf(newname, len, "%s/%s/%s", prefix, name, postfix); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ----------------------------------------------- | 
					
						
							|  |  |  | Welcome to the Asterisk development community! | 
					
						
							|  |  |  | Meet you on the asterisk-dev mailing list.  | 
					
						
							|  |  |  | Subscribe at http://lists.digium.com! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Mark Spencer, Kevin P. Fleming and  | 
					
						
							|  |  |  | the Asterisk.org Development Team |