| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2013-04-23 13:41:32 -07:00
										 |  |  |  * Copyright (c) 2008, 2009, 2010, 2011, 2013 Nicira, Inc. | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-15 15:11:30 -07:00
										 |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * You may obtain a copy of the License at: | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-15 15:11:30 -07:00
										 |  |  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  * limitations under the License. | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <config.h>
 | 
					
						
							| 
									
										
										
										
											2014-03-13 15:28:54 -07:00
										 |  |  | #include <inttypes.h>
 | 
					
						
							| 
									
										
										
										
											2011-05-13 11:55:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | #include "backtrace.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-15 14:10:38 +01:00
										 |  |  | #include "openvswitch/vlog.h"
 | 
					
						
							| 
									
										
										
										
											2014-03-13 15:28:54 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | VLOG_DEFINE_THIS_MODULE(backtrace); | 
					
						
							| 
									
										
										
										
											2011-05-13 11:55:22 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef HAVE_BACKTRACE
 | 
					
						
							|  |  |  | #include <execinfo.h>
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | backtrace_capture(struct backtrace *b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     void *frames[BACKTRACE_MAX_FRAMES]; | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     b->n_frames = backtrace(frames, BACKTRACE_MAX_FRAMES); | 
					
						
							|  |  |  |     for (i = 0; i < b->n_frames; i++) { | 
					
						
							|  |  |  |         b->frames[i] = (uintptr_t) frames[i]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-13 15:28:54 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-23 13:41:32 -07:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-05-13 11:55:22 -07:00
										 |  |  | void | 
					
						
							|  |  |  | backtrace_capture(struct backtrace *backtrace) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-12-16 11:04:14 -08:00
										 |  |  |     backtrace->n_frames = 0; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2011-05-13 11:55:22 -07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-13 15:28:54 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | backtrace_format(const struct backtrace *b, struct ds *ds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (b->n_frames) { | 
					
						
							|  |  |  |         int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ds_put_cstr(ds, " (backtrace:"); | 
					
						
							|  |  |  |         for (i = 0; i < b->n_frames; i++) { | 
					
						
							|  |  |  |             ds_put_format(ds, " 0x%08"PRIxPTR, b->frames[i]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ds_put_cstr(ds, ")"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ds_cstr(ds); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | log_backtrace_at(const char *msg, const char *where) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct backtrace b; | 
					
						
							|  |  |  |     struct ds ds = DS_EMPTY_INITIALIZER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     backtrace_capture(&b); | 
					
						
							|  |  |  |     if (msg) { | 
					
						
							|  |  |  |         ds_put_format(&ds, "%s ", msg); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ds_put_cstr(&ds, where); | 
					
						
							|  |  |  |     VLOG_ERR("%s", backtrace_format(&b, &ds)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ds_destroy(&ds); | 
					
						
							|  |  |  | } |