Online Linux Driver Verification Service (alpha)

Rule violation
Driver: tests-model-32_1--test-doublelock.tar.bz2
Kernel: linux-2.6.37
Verification architecture: x86_64
Rule: Mutex lock/unlock
Error trace
Function bodies
Blocks
  • Others...
    Entry point
    Entry point body
    Function calls
    Initialization function calls
    Function without body calls
    Function stack overflows
    Initialization function bodies
    Returns
    Return values
    Asserts
    Assert conditions
    Identation
    Driver environment initialization
    Driver environment function calls
    Driver environment function bodies
    Model asserts
    Model state changes
    Model returns
    Model function calls
    Model function bodies
    Model function function calls
    Model function function bodies
    Model others
    Function bodies without model function calls
Hide Entry point Hide Entry point body Hide Function calls Show Initialization function calls Hide Function without body calls Hide Function stack overflows Hide Function bodies Show Initialization function bodies Hide Blocks Hide Returns Hide Return values Hide Asserts Hide Assert conditions Hide Identation Show Driver environment initialization Hide Driver environment function calls Hide Driver environment function bodies Hide Model asserts Hide Model state changes Hide Model returns Hide Model function calls Show Model function bodies Hide Model function function calls Hide Model function function bodies Hide Model others Show Function bodies without model function calls
-entry_point();
{
107 LDV_IN_INTERRUPT = 1;
116 +ldv_initialize_FOREACH();
122 +tmp = my_init();
122 assert(tmp == 0);
124 ldv_s_misc_fops_file_operations = 0;
127 tmp___1 = nondet_int() { /* The function body is undefined. */ };
127 assert(tmp___1 != 0);
131 tmp___0 = nondet_int() { /* The function body is undefined. */ };
133 assert(tmp___0 == 0);
136 assert(ldv_s_misc_fops_file_operations == 0);
141 -res_misc_open_1 = misc_open(var_group1 /* inode */, var_group2 /* file */);
{
27 +mutex_lock_my_lock(&(my_lock) /* lock */);
28 -alock();
{
21 +mutex_lock_my_lock(&(my_lock) /* lock */);
}
}
}
Source code
1 2 /** 3 * The test checks that double mutex lock is detected on the models 32_1,32_2 4 */ 5 #include <linux/kernel.h> 6 #include <linux/module.h> 7 #include <linux/mutex.h> 8 #include <linux/major.h> 9 #include <linux/fs.h> 10 11 static DEFINE_MUTEX(my_lock); 12 13 static int misc_open(struct inode * inode, struct file * file); 14 15 static const struct file_operations misc_fops = { 16 .owner = THIS_MODULE, 17 .open = misc_open, 18 }; 19 20 static void alock(void) { 21 mutex_lock(&my_lock); 22 mutex_unlock(&my_lock); 23 } 24 25 static int misc_open(struct inode * inode, struct file * file) 26 { 27 mutex_lock(&my_lock); 28 alock(); 29 mutex_unlock(&my_lock); 30 return 0; 31 } 32 33 static int __init my_init(void) 34 { 35 if (register_chrdev(MISC_MAJOR,"misc",&misc_fops)) 36 goto fail_register; 37 return 0; 38 39 fail_register: 40 return -1; 41 } 42 43 static void __exit my_exit(void) 44 { 45 } 46 47 module_init(my_init); 48 module_exit(my_exit); 49 50 MODULE_LICENSE("Apache 2.0"); 51 MODULE_AUTHOR("LDV Project, Vadim Mutilin <mutilin@ispras.ru>"); 52 53 54 55 56 57 58 /* LDV_COMMENT_BEGIN_MAIN */ 59 #ifdef LDV_MAIN0_sequence_infinite_withcheck_stateful 60 61 /*###########################################################################*/ 62 63 /*############## Driver Environment Generator 0.2 output ####################*/ 64 65 /*###########################################################################*/ 66 67 68 69 /* LDV_COMMENT_FUNCTION_DECLARE_LDV Special function for LDV verifier. Test if all kernel resources are correctly released by driver before driver will be unloaded. */ 70 void ldv_check_final_state(void); 71 72 /* LDV_COMMENT_FUNCTION_DECLARE_LDV Special function for LDV verifier. Test correct return result. */ 73 void ldv_check_return_value(int res); 74 75 /* LDV_COMMENT_FUNCTION_DECLARE_LDV Special function for LDV verifier. Initializes the model. */ 76 void ldv_initialize(void); 77 78 /* LDV_COMMENT_FUNCTION_DECLARE_LDV Special function for LDV verifier. Returns arbitrary interger value. */ 79 int nondet_int(void); 80 81 /* LDV_COMMENT_VAR_DECLARE_LDV Special variable for LDV verifier. */ 82 int LDV_IN_INTERRUPT; 83 84 /* LDV_COMMENT_FUNCTION_MAIN Main function for LDV verifier. */ 85 void ldv_main0_sequence_infinite_withcheck_stateful(void) { 86 87 88 89 /* LDV_COMMENT_BEGIN_VARIABLE_DECLARATION_PART */ 90 /*============================= VARIABLE DECLARATION PART =============================*/ 91 /** STRUCT: struct type: file_operations, struct name: misc_fops **/ 92 /* content: static int misc_open(struct inode * inode, struct file * file)*/ 93 /* LDV_COMMENT_END_PREP */ 94 /* LDV_COMMENT_VAR_DECLARE Variable declaration for function "misc_open" */ 95 struct inode * var_group1; 96 /* LDV_COMMENT_VAR_DECLARE Variable declaration for function "misc_open" */ 97 struct file * var_group2; 98 /* LDV_COMMENT_VAR_DECLARE Variable declaration for test return result from function call "misc_open" */ 99 static int res_misc_open_1; 100 101 102 103 104 /* LDV_COMMENT_END_VARIABLE_DECLARATION_PART */ 105 /* LDV_COMMENT_BEGIN_VARIABLE_INITIALIZING_PART */ 106 /*============================= VARIABLE INITIALIZING PART =============================*/ 107 LDV_IN_INTERRUPT=1; 108 109 110 111 112 /* LDV_COMMENT_END_VARIABLE_INITIALIZING_PART */ 113 /* LDV_COMMENT_BEGIN_FUNCTION_CALL_SECTION */ 114 /*============================= FUNCTION CALL SECTION =============================*/ 115 /* LDV_COMMENT_FUNCTION_CALL Initialize LDV model. */ 116 ldv_initialize(); 117 118 /** INIT: init_type: ST_MODULE_INIT **/ 119 /* content: static int __init my_init(void)*/ 120 /* LDV_COMMENT_END_PREP */ 121 /* LDV_COMMENT_FUNCTION_CALL Kernel calls driver init function after driver loading to kernel. This function declared as "MODULE_INIT(function name)". */ 122 if(my_init()) 123 goto ldv_final; 124 int ldv_s_misc_fops_file_operations = 0; 125 126 127 while( nondet_int() 128 || !(ldv_s_misc_fops_file_operations == 0) 129 ) { 130 131 switch(nondet_int()) { 132 133 case 0: { 134 135 /** STRUCT: struct type: file_operations, struct name: misc_fops **/ 136 if(ldv_s_misc_fops_file_operations==0) { 137 138 /* content: static int misc_open(struct inode * inode, struct file * file)*/ 139 /* LDV_COMMENT_END_PREP */ 140 /* LDV_COMMENT_FUNCTION_CALL Function from field "open" from driver structure with callbacks "misc_fops". Standart function test for correct return result. */ 141 res_misc_open_1 = misc_open( var_group1, var_group2); 142 ldv_check_return_value(res_misc_open_1); 143 if(res_misc_open_1) 144 goto ldv_module_exit; 145 ldv_s_misc_fops_file_operations=0; 146 147 } 148 149 } 150 151 break; 152 default: break; 153 154 } 155 156 } 157 158 ldv_module_exit: 159 160 /** INIT: init_type: ST_MODULE_EXIT **/ 161 /* content: static void __exit my_exit(void)*/ 162 /* LDV_COMMENT_END_PREP */ 163 /* LDV_COMMENT_FUNCTION_CALL Kernel calls driver release function before driver will be uploaded from kernel. This function declared as "MODULE_EXIT(function name)". */ 164 my_exit(); 165 166 /* LDV_COMMENT_FUNCTION_CALL Checks that all resources and locks are correctly released before the driver will be unloaded. */ 167 ldv_final: ldv_check_final_state(); 168 169 /* LDV_COMMENT_END_FUNCTION_CALL_SECTION */ 170 return; 171 172 } 173 #endif 174 175 /* LDV_COMMENT_END_MAIN */ 176 /* LDV_COMMENT_BEGIN_MODEL */ 177 178 /* 179 * CONFIG_DEBUG_LOCK_ALLOC must be turned off to apply the given model. 180 * To be independ on the value of this flag there is the corresponding 181 * aspect model. 182 */ 183 184 /* the function works only without aspectator */ 185 long __builtin_expect(long val, long res) { 186 return val; 187 } 188 189 #include "engine-blast.h" 190 191 #include <linux/kernel.h> 192 #include <linux/mutex.h> 193 194 /* Need this because rerouter is buggy!.. */ 195 extern int ldv_mutex_TEMPLATE; 196 /* Now the actual variable goes... */ 197 int ldv_mutex_TEMPLATE = 1; 198 199 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='mutex_lock_interruptible') Check that the mutex was unlocked and nondeterministically lock it. Return the corresponding error code on fails*/ 200 int mutex_lock_interruptible_TEMPLATE(struct mutex *lock) 201 { 202 int nondetermined; 203 204 /* LDV_COMMENT_ASSERT Mutex must be unlocked*/ 205 ldv_assert(ldv_mutex_TEMPLATE == 1); 206 207 /* LDV_COMMENT_OTHER Construct the nondetermined result*/ 208 nondetermined = ldv_undef_int(); 209 210 /* LDV_COMMENT_ASSERT Nondeterministically lock the mutex*/ 211 if (nondetermined) 212 { 213 /* LDV_COMMENT_CHANGE_STATE Lock the mutex*/ 214 ldv_mutex_TEMPLATE = 2; 215 /* LDV_COMMENT_RETURN Finish with success*/ 216 return 0; 217 } 218 else 219 { 220 /* LDV_COMMENT_RETURN Finish with the fail. The mutex is keeped unlocked*/ 221 return -EINTR; 222 } 223 } 224 225 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='atomic_dec_and_mutex_lock') Lock if atomic decrement result is zero */ 226 int atomic_dec_and_mutex_lock_TEMPLATE(atomic_t *cnt, struct mutex *lock) 227 { 228 int atomic_value_after_dec; 229 230 /* LDV_COMMENT_ASSERT Mutex must be unlocked (since we may lock it in this function) */ 231 ldv_assert(ldv_mutex_TEMPLATE == 1); 232 233 /* LDV_COMMENT_OTHER Assign the result of atomic decrement */ 234 atomic_value_after_dec = ldv_undef_int(); 235 236 /* LDV_COMMENT_ASSERT Check if atomic decrement returns zero */ 237 if (atomic_value_after_dec == 0) 238 { 239 /* LDV_COMMENT_CHANGE_STATE Lock the mutex, as atomic has decremented to zero*/ 240 ldv_mutex_TEMPLATE = 2; 241 /* LDV_COMMENT_RETURN Return 1 with a locked mutex */ 242 return 1; 243 } 244 245 /* LDV_COMMENT_RETURN Atomic decrement is still not zero, return 0 without locking the mutex */ 246 return 0; 247 } 248 249 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='mutex_lock_killable') Check that the mutex wasn unlocked and nondeterministically lock it. Return the corresponding error code on fails*/ 250 int __must_check mutex_lock_killable_TEMPLATE(struct mutex *lock) 251 { 252 int nondetermined; 253 254 /* LDV_COMMENT_ASSERT Mutex must be unlocked*/ 255 ldv_assert(ldv_mutex_TEMPLATE == 1); 256 257 /* LDV_COMMENT_OTHER Construct the nondetermined result*/ 258 nondetermined = ldv_undef_int(); 259 260 /* LDV_COMMENT_ASSERT Nondeterministically lock the mutex*/ 261 if (nondetermined) 262 { 263 /* LDV_COMMENT_CHANGE_STATE Lock the mutex*/ 264 ldv_mutex_TEMPLATE = 2; 265 /* LDV_COMMENT_RETURN Finish with success*/ 266 return 0; 267 } 268 else 269 { 270 /* LDV_COMMENT_RETURN Finish with the fail. The mutex is keeped unlocked*/ 271 return -EINTR; 272 } 273 } 274 275 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='mutex_is_locked') Checks whether the mutex is locked*/ 276 int mutex_is_locked_TEMPLATE(struct mutex *lock) 277 { 278 int nondetermined; 279 280 if(ldv_mutex_TEMPLATE == 1) { 281 /* LDV_COMMENT_OTHER Construct the nondetermined result*/ 282 nondetermined = ldv_undef_int(); 283 if(nondetermined) { 284 /* LDV_COMMENT_RETURN the mutex is unlocked*/ 285 return 0; 286 } else { 287 /* LDV_COMMENT_RETURN the mutex is locked*/ 288 return 1; 289 } 290 } else { 291 /* LDV_COMMENT_RETURN the mutex is locked*/ 292 return 1; 293 } 294 } 295 296 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='mutex_lock(?!_interruptible|_killable)') Check that the mutex was not locked and lock it*/ 297 void mutex_lock_TEMPLATE(struct mutex *lock) 298 { 299 /* LDV_COMMENT_ASSERT Mutex must be unlocked*/ 300 ldv_assert(ldv_mutex_TEMPLATE == 1); 301 /* LDV_COMMENT_CHANGE_STATE Lock the mutex*/ 302 ldv_mutex_TEMPLATE = 2; 303 } 304 305 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='mutex_trylock') Check that the mutex was not locked and nondeterministically lock it. Return 0 on fails*/ 306 int mutex_trylock_TEMPLATE(struct mutex *lock) 307 { 308 int is_mutex_held_by_another_thread; 309 310 /* LDV_COMMENT_ASSERT Mutex must be unlocked*/ 311 ldv_assert(ldv_mutex_TEMPLATE == 1); 312 313 /* LDV_COMMENT_OTHER Construct the nondetermined result*/ 314 is_mutex_held_by_another_thread = ldv_undef_int(); 315 316 /* LDV_COMMENT_ASSERT Nondeterministically lock the mutex*/ 317 if (is_mutex_held_by_another_thread) 318 { 319 /* LDV_COMMENT_RETURN Finish with fail*/ 320 return 0; 321 } 322 else 323 { 324 /* LDV_COMMENT_CHANGE_STATE Lock the mutex*/ 325 ldv_mutex_TEMPLATE = 2; 326 /* LDV_COMMENT_RETURN Finish with success*/ 327 return 1; 328 } 329 } 330 331 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='mutex_unlock') Check that the mutex was locked and unlock it*/ 332 void mutex_unlock_TEMPLATE(struct mutex *lock) 333 { 334 /* LDV_COMMENT_ASSERT Mutex must be locked*/ 335 ldv_assert(ldv_mutex_TEMPLATE == 2); 336 /* LDV_COMMENT_CHANGE_STATE Unlock the mutex*/ 337 ldv_mutex_TEMPLATE = 1; 338 } 339 340 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='check_final_state') Check that the mutex is unlocked at the end*/ 341 void ldv_check_final_state_TEMPLATE(void) 342 { 343 /* LDV_COMMENT_ASSERT The mutex must be unlocked at the end*/ 344 ldv_assert(ldv_mutex_TEMPLATE == 1); 345 } 346 347 /* LDV_COMMENT_MODEL_FUNCTION_DEFINITION(name='ldv_initialize') Initialize mutex variables*/ 348 void ldv_initialize_TEMPLATE(void) 349 { 350 /* LDV_COMMENT_ASSERT Initialize mutex with initial model value*/ 351 ldv_mutex_TEMPLATE = 1; 352 } 353 354 /* LDV_COMMENT_END_MODEL */

Here is the explanation of the rule violation arisen in your driver for the corresponding kernel.

Note that there may be no error indeed. Please see on error trace and source code to understand whether there is an error in your driver.

The Error trace column contains the path on which rule is violated. You can choose some entity classes to be shown or hiden by clicking on the corresponding checkboxes or in the advanced Others menu. Also you can show or hide each particular entity by clicking on the corresponding - or +. In hovering on some entities you can see their descriptions and meaning. Also the error trace is binded with the source code. Line numbers are shown as links on the left. You can click on them to open the corresponding line in source code. Line numbers and file names are shown in entity descriptions.

The Source code column contains content of files related with the error trace. There are your driver (note that there are some our modifications at the end), kernel headers and rule source code. Tabs show the currently opened file and other available files. In hovering you can see file names in titles. On clicking the corresponding file content will be shown.