view usr/src/tools/smatch/src/check_locking.c @ 19241:79022555a4a9

11972 resync smatch Reviewed by: Robert Mustacchi <rm@fingolfin.org> Approved by: Dan McDonald <danmcd@joyent.com>
author John Levon <john.levon@joyent.com>
date Mon, 11 Nov 2019 16:23:50 +0000
parents 8759ad1a3e4a
children 93e34922a8e2
line wrap: on
line source

/*
 * Copyright (C) 2009 Dan Carpenter.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, see http://www.gnu.org/copyleft/gpl.txt
 */

/*
 * This test checks that locks are held the same across all returns.
 *
 * Of course, some functions are designed to only hold the locks on success.
 * Oh well... We can rewrite it later if we want.
 *
 * The list of wine locking functions came from an earlier script written
 * by Michael Stefaniuc.
 *
 */

#include "parse.h"
#include "smatch.h"
#include "smatch_extra.h"
#include "smatch_slist.h"

static int my_id;

static int func_has_transition;

STATE(locked);
STATE(start_state);
STATE(unlocked);
STATE(impossible);

enum action {
	LOCK,
	UNLOCK,
};

enum return_type {
	ret_any,
	ret_non_zero,
	ret_zero,
	ret_one,
	ret_negative,
	ret_positive,
};

#define RETURN_VAL -1
#define NO_ARG -2

struct lock_info {
	const char *function;
	enum action action;
	const char *name;
	int arg;
	enum return_type return_type;
};

static struct lock_info wine_lock_table[] = {
	{"create_window_handle", LOCK, "create_window_handle", RETURN_VAL, ret_non_zero},
	{"WIN_GetPtr", LOCK, "create_window_handle", RETURN_VAL, ret_non_zero},
	{"WIN_ReleasePtr", UNLOCK, "create_window_handle", 0, ret_any},
	{"EnterCriticalSection", LOCK, "CriticalSection", 0, ret_any},
	{"LeaveCriticalSection", UNLOCK, "CriticalSection", 0, ret_any},
	{"RtlEnterCriticalSection", LOCK, "RtlCriticalSection", 0, ret_any},
	{"RtlLeaveCriticalSection", UNLOCK, "RtlCriticalSection", 0, ret_any},
	{"GDI_GetObjPtr", LOCK, "GDI_Get", 0, ret_non_zero},
	{"GDI_ReleaseObj", UNLOCK, "GDI_Get", 0, ret_any},
	{"LdrLockLoaderLock", LOCK, "LdrLockLoaderLock", 2, ret_any},
	{"LdrUnlockLoaderLock", UNLOCK, "LdrLockLoaderLock", 1, ret_any},
	{"_lock", LOCK, "_lock", 0, ret_any},
	{"_unlock", UNLOCK, "_lock", 0, ret_any},
	{"msiobj_lock", LOCK, "msiobj_lock", 0, ret_any},
	{"msiobj_unlock", UNLOCK, "msiobj_lock", 0, ret_any},
	{"RtlAcquirePebLock", LOCK, "PebLock", NO_ARG, ret_any},
	{"RtlReleasePebLock", UNLOCK, "PebLock", NO_ARG, ret_any},
	{"server_enter_uninterrupted_section", LOCK, "server_uninterrupted_section", 0, ret_any},
	{"server_leave_uninterrupted_section", UNLOCK, "server_uninterrupted_section", 0, ret_any},
	{"RtlLockHeap", LOCK, "RtlLockHeap", 0, ret_any},
	{"RtlUnlockHeap", UNLOCK, "RtlLockHeap", 0, ret_any},
	{"_EnterSysLevel", LOCK, "SysLevel", 0, ret_any},
	{"_LeaveSysLevel", UNLOCK, "SysLevel", 0, ret_any},
	{"USER_Lock", LOCK, "USER_Lock", NO_ARG, ret_any},
	{"USER_Unlock", UNLOCK, "USER_Lock", NO_ARG, ret_any},
	{"wine_tsx11_lock", LOCK, "wine_tsx11_lock", NO_ARG, ret_any},
	{"wine_tsx11_unlock", UNLOCK, "wine_tsx11_lock", NO_ARG, ret_any},
	{"wine_tsx11_lock_ptr", LOCK, "wine_tsx11_lock_ptr", NO_ARG, ret_any},
	{"wine_tsx11_unlock_ptr", UNLOCK, "wine_tsx11_lock_ptr", NO_ARG, ret_any},
	{"wined3d_mutex_lock", LOCK, "wined3d_mutex_lock", NO_ARG, ret_any},
	{"wined3d_mutex_unlock", UNLOCK, "wined3d_mutex_lock", NO_ARG, ret_any},
	{"X11DRV_DIB_Lock", LOCK, "X11DRV_DIB_Lock", 0, ret_any},
	{"X11DRV_DIB_Unlock", UNLOCK, "X11DRV_DIB_Lock", 0, ret_any},
};

static struct lock_info kernel_lock_table[] = {
	{"lock_kernel",   LOCK,   "BKL", NO_ARG, ret_any},
	{"unlock_kernel", UNLOCK, "BKL", NO_ARG, ret_any},

	{"spin_lock",                  LOCK,   "spin_lock", 0, ret_any},
	{"spin_unlock",                UNLOCK, "spin_lock", 0, ret_any},
	{"spin_lock_nested",           LOCK,   "spin_lock", 0, ret_any},
	{"_spin_lock",                 LOCK,   "spin_lock", 0, ret_any},
	{"_spin_unlock",               UNLOCK, "spin_lock", 0, ret_any},
	{"_spin_lock_nested",          LOCK,   "spin_lock", 0, ret_any},
	{"__spin_lock",                LOCK,   "spin_lock", 0, ret_any},
	{"__spin_unlock",              UNLOCK, "spin_lock", 0, ret_any},
	{"__spin_lock_nested",         LOCK,   "spin_lock", 0, ret_any},
	{"raw_spin_lock",              LOCK,   "spin_lock", 0, ret_any},
	{"raw_spin_unlock",            UNLOCK, "spin_lock", 0, ret_any},
	{"_raw_spin_lock",             LOCK,   "spin_lock", 0, ret_any},
	{"_raw_spin_lock_nested",      LOCK,   "spin_lock", 0, ret_any},
	{"_raw_spin_unlock",           UNLOCK, "spin_lock", 0, ret_any},
	{"__raw_spin_lock",            LOCK,   "spin_lock", 0, ret_any},
	{"__raw_spin_unlock",          UNLOCK, "spin_lock", 0, ret_any},

	{"spin_lock_irq",              LOCK,   "spin_lock", 0, ret_any},
	{"spin_unlock_irq",            UNLOCK, "spin_lock", 0, ret_any},
	{"_spin_lock_irq",             LOCK,   "spin_lock", 0, ret_any},
	{"_spin_unlock_irq",           UNLOCK, "spin_lock", 0, ret_any},
	{"__spin_lock_irq",            LOCK,   "spin_lock", 0, ret_any},
	{"__spin_unlock_irq",          UNLOCK, "spin_lock", 0, ret_any},
	{"_raw_spin_lock_irq",         LOCK,   "spin_lock", 0, ret_any},
	{"_raw_spin_unlock_irq",       UNLOCK, "spin_lock", 0, ret_any},
	{"__raw_spin_unlock_irq",      UNLOCK, "spin_lock", 0, ret_any},
	{"spin_lock_irqsave",          LOCK,   "spin_lock", 0, ret_any},
	{"spin_unlock_irqrestore",     UNLOCK, "spin_lock", 0, ret_any},
	{"_spin_lock_irqsave",         LOCK,   "spin_lock", 0, ret_any},
	{"_spin_unlock_irqrestore",    UNLOCK, "spin_lock", 0, ret_any},
	{"__spin_lock_irqsave",        LOCK,   "spin_lock", 0, ret_any},
	{"__spin_unlock_irqrestore",   UNLOCK, "spin_lock", 0, ret_any},
	{"_raw_spin_lock_irqsave",     LOCK,   "spin_lock", 0, ret_any},
	{"_raw_spin_unlock_irqrestore", UNLOCK, "spin_lock", 0, ret_any},
	{"__raw_spin_lock_irqsave",    LOCK,   "spin_lock", 0, ret_any},
	{"__raw_spin_unlock_irqrestore", UNLOCK, "spin_lock", 0, ret_any},
	{"spin_lock_irqsave_nested",   LOCK,   "spin_lock", 0, ret_any},
	{"_spin_lock_irqsave_nested",  LOCK,   "spin_lock", 0, ret_any},
	{"__spin_lock_irqsave_nested", LOCK,   "spin_lock", 0, ret_any},
	{"_raw_spin_lock_irqsave_nested", LOCK, "spin_lock", 0, ret_any},
	{"spin_lock_bh",               LOCK,   "spin_lock", 0, ret_any},
	{"spin_unlock_bh",             UNLOCK, "spin_lock", 0, ret_any},
	{"_spin_lock_bh",              LOCK,   "spin_lock", 0, ret_any},
	{"_spin_unlock_bh",            UNLOCK, "spin_lock", 0, ret_any},
	{"__spin_lock_bh",             LOCK,   "spin_lock", 0, ret_any},
	{"__spin_unlock_bh",           UNLOCK, "spin_lock", 0, ret_any},

	{"spin_trylock",               LOCK,   "spin_lock", 0, ret_one},
	{"_spin_trylock",              LOCK,   "spin_lock", 0, ret_one},
	{"__spin_trylock",             LOCK,   "spin_lock", 0, ret_one},
	{"raw_spin_trylock",           LOCK,   "spin_lock", 0, ret_one},
	{"_raw_spin_trylock",          LOCK,   "spin_lock", 0, ret_one},
	{"spin_trylock_irq",           LOCK,   "spin_lock", 0, ret_one},
	{"spin_trylock_irqsave",       LOCK,   "spin_lock", 0, ret_one},
	{"spin_trylock_bh",            LOCK,   "spin_lock", 0, ret_one},
	{"_spin_trylock_bh",           LOCK,   "spin_lock", 0, ret_one},
	{"__spin_trylock_bh",          LOCK,   "spin_lock", 0, ret_one},
	{"__raw_spin_trylock",         LOCK,   "spin_lock", 0, ret_one},
	{"_atomic_dec_and_lock",       LOCK,   "spin_lock", 1, ret_one},

	{"read_lock",                 LOCK,   "read_lock", 0, ret_any},
	{"read_unlock",               UNLOCK, "read_lock", 0, ret_any},
	{"_read_lock",                LOCK,   "read_lock", 0, ret_any},
	{"_read_unlock",              UNLOCK, "read_lock", 0, ret_any},
	{"__read_lock",               LOCK,   "read_lock", 0, ret_any},
	{"__read_unlock",             UNLOCK, "read_lock", 0, ret_any},
	{"_raw_read_lock",            LOCK,   "read_lock", 0, ret_any},
	{"_raw_read_unlock",          UNLOCK, "read_lock", 0, ret_any},
	{"__raw_read_lock",           LOCK,   "read_lock", 0, ret_any},
	{"__raw_read_unlock",         UNLOCK, "read_lock", 0, ret_any},
	{"read_lock_irq",             LOCK,   "read_lock", 0, ret_any},
	{"read_unlock_irq" ,          UNLOCK, "read_lock", 0, ret_any},
	{"_read_lock_irq",            LOCK,   "read_lock", 0, ret_any},
	{"_read_unlock_irq",          UNLOCK, "read_lock", 0, ret_any},
	{"__read_lock_irq",           LOCK,   "read_lock", 0, ret_any},
	{"__read_unlock_irq",         UNLOCK, "read_lock", 0, ret_any},
	{"read_lock_irqsave",         LOCK,   "read_lock", 0, ret_any},
	{"read_unlock_irqrestore",    UNLOCK, "read_lock", 0, ret_any},
	{"_read_lock_irqsave",        LOCK,   "read_lock", 0, ret_any},
	{"_read_unlock_irqrestore",   UNLOCK, "read_lock", 0, ret_any},
	{"__read_lock_irqsave",       LOCK,   "read_lock", 0, ret_any},
	{"__read_unlock_irqrestore",  UNLOCK, "read_lock", 0, ret_any},
	{"read_lock_bh",              LOCK,   "read_lock", 0, ret_any},
	{"read_unlock_bh",            UNLOCK, "read_lock", 0, ret_any},
	{"_read_lock_bh",             LOCK,   "read_lock", 0, ret_any},
	{"_read_unlock_bh",           UNLOCK, "read_lock", 0, ret_any},
	{"__read_lock_bh",            LOCK,   "read_lock", 0, ret_any},
	{"__read_unlock_bh",          UNLOCK, "read_lock", 0, ret_any},
	{"_raw_read_lock_bh",         LOCK,   "read_lock", 0, ret_any},
	{"_raw_read_unlock_bh",       UNLOCK, "read_lock", 0, ret_any},
	{"__raw_read_lock_bh",        LOCK,   "read_lock", 0, ret_any},
	{"__raw_read_unlock_bh",      UNLOCK, "read_lock", 0, ret_any},

	{"generic__raw_read_trylock", LOCK,   "read_lock", 0, ret_one},
	{"read_trylock",              LOCK,   "read_lock", 0, ret_one},
	{"_read_trylock",             LOCK,   "read_lock", 0, ret_one},
	{"raw_read_trylock",          LOCK,   "read_lock", 0, ret_one},
	{"_raw_read_trylock",         LOCK,   "read_lock", 0, ret_one},
	{"__raw_read_trylock",        LOCK,   "read_lock", 0, ret_one},
	{"__read_trylock",            LOCK,   "read_lock", 0, ret_one},

	{"write_lock",                LOCK,   "write_lock", 0, ret_any},
	{"write_unlock",              UNLOCK, "write_lock", 0, ret_any},
	{"_write_lock",               LOCK,   "write_lock", 0, ret_any},
	{"_write_unlock",             UNLOCK, "write_lock", 0, ret_any},
	{"__write_lock",              LOCK,   "write_lock", 0, ret_any},
	{"__write_unlock",            UNLOCK, "write_lock", 0, ret_any},
	{"write_lock_irq",            LOCK,   "write_lock", 0, ret_any},
	{"write_unlock_irq",          UNLOCK, "write_lock", 0, ret_any},
	{"_write_lock_irq",           LOCK,   "write_lock", 0, ret_any},
	{"_write_unlock_irq",         UNLOCK, "write_lock", 0, ret_any},
	{"__write_lock_irq",          LOCK,   "write_lock", 0, ret_any},
	{"__write_unlock_irq",        UNLOCK, "write_lock", 0, ret_any},
	{"write_lock_irqsave",        LOCK,   "write_lock", 0, ret_any},
	{"write_unlock_irqrestore",   UNLOCK, "write_lock", 0, ret_any},
	{"_write_lock_irqsave",       LOCK,   "write_lock", 0, ret_any},
	{"_write_unlock_irqrestore",  UNLOCK, "write_lock", 0, ret_any},
	{"__write_lock_irqsave",      LOCK,   "write_lock", 0, ret_any},
	{"__write_unlock_irqrestore", UNLOCK, "write_lock", 0, ret_any},
	{"write_lock_bh",             LOCK,   "write_lock", 0, ret_any},
	{"write_unlock_bh",           UNLOCK, "write_lock", 0, ret_any},
	{"_write_lock_bh",            LOCK,   "write_lock", 0, ret_any},
	{"_write_unlock_bh",          UNLOCK, "write_lock", 0, ret_any},
	{"__write_lock_bh",           LOCK,   "write_lock", 0, ret_any},
	{"__write_unlock_bh",         UNLOCK, "write_lock", 0, ret_any},
	{"_raw_write_lock",           LOCK,   "write_lock", 0, ret_any},
	{"__raw_write_lock",          LOCK,   "write_lock", 0, ret_any},
	{"_raw_write_unlock",         UNLOCK, "write_lock", 0, ret_any},
	{"__raw_write_unlock",        UNLOCK, "write_lock", 0, ret_any},

	{"write_trylock",             LOCK,   "write_lock", 0, ret_one},
	{"_write_trylock",            LOCK,   "write_lock", 0, ret_one},
	{"raw_write_trylock",         LOCK,   "write_lock", 0, ret_one},
	{"_raw_write_trylock",        LOCK,   "write_lock", 0, ret_one},
	{"__write_trylock",           LOCK,   "write_lock", 0, ret_one},
	{"__raw_write_trylock",       LOCK,   "write_lock", 0, ret_one},

	{"down",               LOCK,   "sem", 0, ret_any},
	{"up",                 UNLOCK, "sem", 0, ret_any},
	{"down_trylock",       LOCK,   "sem", 0, ret_zero},
	{"down_timeout",       LOCK,   "sem", 0, ret_zero},
	{"down_interruptible", LOCK,   "sem", 0, ret_zero},


	{"down_write",          LOCK,   "rw_sem", 0, ret_any},
	{"downgrade_write",     UNLOCK, "rw_sem", 0, ret_any},
	{"downgrade_write",     LOCK,   "read_sem", 0, ret_any},
	{"up_write",            UNLOCK, "rw_sem", 0, ret_any},
	{"down_write_trylock",  LOCK,   "rw_sem", 0, ret_one},
	{"down_write_killable", LOCK,   "rw_sem", 0, ret_zero},
	{"down_read",           LOCK,   "read_sem", 0, ret_any},
	{"down_read_trylock",   LOCK,   "read_sem", 0, ret_one},
	{"down_read_killable",  LOCK,   "read_sem", 0, ret_zero},
	{"up_read",             UNLOCK, "read_sem", 0, ret_any},

	{"mutex_lock",                      LOCK,   "mutex", 0, ret_any},
	{"mutex_lock_io",                   LOCK,   "mutex", 0, ret_any},
	{"mutex_unlock",                    UNLOCK, "mutex", 0, ret_any},
	{"mutex_lock_nested",               LOCK,   "mutex", 0, ret_any},
	{"mutex_lock_io_nested",            LOCK,   "mutex", 0, ret_any},

	{"mutex_lock_interruptible",        LOCK,   "mutex", 0, ret_zero},
	{"mutex_lock_interruptible_nested", LOCK,   "mutex", 0, ret_zero},
	{"mutex_lock_killable",             LOCK,   "mutex", 0, ret_zero},
	{"mutex_lock_killable_nested",      LOCK,   "mutex", 0, ret_zero},

	{"mutex_trylock",                   LOCK,   "mutex", 0, ret_one},

	{"raw_local_irq_disable", LOCK,   "irq", NO_ARG, ret_any},
	{"raw_local_irq_enable",  UNLOCK, "irq", NO_ARG, ret_any},
	{"spin_lock_irq",         LOCK,   "irq", NO_ARG, ret_any},
	{"spin_unlock_irq",       UNLOCK, "irq", NO_ARG, ret_any},
	{"_spin_lock_irq",        LOCK,   "irq", NO_ARG, ret_any},
	{"_spin_unlock_irq",      UNLOCK, "irq", NO_ARG, ret_any},
	{"__spin_lock_irq",       LOCK,   "irq", NO_ARG, ret_any},
	{"__spin_unlock_irq",     UNLOCK, "irq", NO_ARG, ret_any},
	{"_raw_spin_lock_irq",    LOCK,   "irq", NO_ARG, ret_any},
	{"_raw_spin_unlock_irq",  UNLOCK, "irq", NO_ARG, ret_any},
	{"__raw_spin_unlock_irq", UNLOCK, "irq", NO_ARG, ret_any},
	{"spin_trylock_irq",      LOCK,   "irq", NO_ARG, ret_one},
	{"read_lock_irq",         LOCK,   "irq", NO_ARG, ret_any},
	{"read_unlock_irq",       UNLOCK, "irq", NO_ARG, ret_any},
	{"_read_lock_irq",        LOCK,   "irq", NO_ARG, ret_any},
	{"_read_unlock_irq",      UNLOCK, "irq", NO_ARG, ret_any},
	{"__read_lock_irq",       LOCK,   "irq", NO_ARG, ret_any},
	{"__read_unlock_irq",     UNLOCK, "irq", NO_ARG, ret_any},
	{"write_lock_irq",        LOCK,   "irq", NO_ARG, ret_any},
	{"write_unlock_irq",      UNLOCK, "irq", NO_ARG, ret_any},
	{"_write_lock_irq",       LOCK,   "irq", NO_ARG, ret_any},
	{"_write_unlock_irq",     UNLOCK, "irq", NO_ARG, ret_any},
	{"__write_lock_irq",      LOCK,   "irq", NO_ARG, ret_any},
	{"__write_unlock_irq",    UNLOCK, "irq", NO_ARG, ret_any},

	{"arch_local_irq_save",        LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"arch_local_irq_restore",     UNLOCK, "irqsave", 0, ret_any},
	{"__raw_local_irq_save",       LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"raw_local_irq_restore",      UNLOCK, "irqsave", 0, ret_any},
	{"spin_lock_irqsave_nested",   LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"spin_lock_irqsave",          LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"spin_lock_irqsave",          LOCK,   "irqsave", 1, ret_any},
	{"spin_unlock_irqrestore",     UNLOCK, "irqsave", 1, ret_any},
	{"_spin_lock_irqsave_nested",  LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"_spin_lock_irqsave",         LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"_spin_lock_irqsave",         LOCK,   "irqsave", 1, ret_any},
	{"_spin_unlock_irqrestore",    UNLOCK, "irqsave", 1, ret_any},
	{"__spin_lock_irqsave_nested", LOCK,   "irqsave", 1, ret_any},
	{"__spin_lock_irqsave",        LOCK,   "irqsave", 1, ret_any},
	{"__spin_unlock_irqrestore",   UNLOCK, "irqsave", 1, ret_any},
	{"_raw_spin_lock_irqsave",     LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"_raw_spin_lock_irqsave",     LOCK,   "irqsave", 1, ret_any},
	{"_raw_spin_unlock_irqrestore",UNLOCK, "irqsave", 1, ret_any},
	{"__raw_spin_lock_irqsave",    LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"__raw_spin_unlock_irqrestore",UNLOCK, "irqsave", 1, ret_any},
	{"_raw_spin_lock_irqsave_nested", LOCK, "irqsave", RETURN_VAL, ret_any},
	{"spin_trylock_irqsave",       LOCK,   "irqsave", 1, ret_one},
	{"read_lock_irqsave",          LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"read_lock_irqsave",          LOCK,   "irqsave", 1, ret_any},
	{"read_unlock_irqrestore",     UNLOCK, "irqsave", 1, ret_any},
	{"_read_lock_irqsave",         LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"_read_lock_irqsave",         LOCK,   "irqsave", 1, ret_any},
	{"_read_unlock_irqrestore",    UNLOCK, "irqsave", 1, ret_any},
	{"__read_lock_irqsave",        LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"__read_unlock_irqrestore",   UNLOCK, "irqsave", 1, ret_any},
	{"write_lock_irqsave",         LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"write_lock_irqsave",         LOCK,   "irqsave", 1, ret_any},
	{"write_unlock_irqrestore",    UNLOCK, "irqsave", 1, ret_any},
	{"_write_lock_irqsave",        LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"_write_lock_irqsave",        LOCK,   "irqsave", 1, ret_any},
	{"_write_unlock_irqrestore",   UNLOCK, "irqsave", 1, ret_any},
	{"__write_lock_irqsave",       LOCK,   "irqsave", RETURN_VAL, ret_any},
	{"__write_unlock_irqrestore",  UNLOCK, "irqsave", 1, ret_any},

	{"local_bh_disable",	LOCK,	"bottom_half", NO_ARG, ret_any},
	{"_local_bh_disable",	LOCK,	"bottom_half", NO_ARG, ret_any},
	{"__local_bh_disable",	LOCK,	"bottom_half", NO_ARG, ret_any},
	{"local_bh_enable",	UNLOCK,	"bottom_half", NO_ARG, ret_any},
	{"_local_bh_enable",	UNLOCK,	"bottom_half", NO_ARG, ret_any},
	{"__local_bh_enable",	UNLOCK,	"bottom_half", NO_ARG, ret_any},
	{"spin_lock_bh",        LOCK,   "bottom_half", NO_ARG, ret_any},
	{"spin_unlock_bh",      UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"_spin_lock_bh",       LOCK,   "bottom_half", NO_ARG, ret_any},
	{"_spin_unlock_bh",     UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"__spin_lock_bh",      LOCK,   "bottom_half", NO_ARG, ret_any},
	{"__spin_unlock_bh",    UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"read_lock_bh",        LOCK,   "bottom_half", NO_ARG, ret_any},
	{"read_unlock_bh",      UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"_read_lock_bh",       LOCK,   "bottom_half", NO_ARG, ret_any},
	{"_read_unlock_bh",     UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"__read_lock_bh",      LOCK,   "bottom_half", NO_ARG, ret_any},
	{"__read_unlock_bh",    UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"_raw_read_lock_bh",   LOCK,   "bottom_half", NO_ARG, ret_any},
	{"_raw_read_unlock_bh", UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"write_lock_bh",       LOCK,   "bottom_half", NO_ARG, ret_any},
	{"write_unlock_bh",     UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"_write_lock_bh",      LOCK,   "bottom_half", NO_ARG, ret_any},
	{"_write_unlock_bh",    UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"__write_lock_bh",     LOCK,   "bottom_half", NO_ARG, ret_any},
	{"__write_unlock_bh",   UNLOCK, "bottom_half", NO_ARG, ret_any},
	{"spin_trylock_bh",     LOCK,   "bottom_half", NO_ARG, ret_one},
	{"_spin_trylock_bh",    LOCK,   "bottom_half", NO_ARG, ret_one},
	{"__spin_trylock_bh",   LOCK,   "bottom_half", NO_ARG, ret_one},

	{"ffs_mutex_lock",        LOCK,   "mutex", 0, ret_zero},
};

static struct lock_info *lock_table;

static struct tracker_list *starts_locked;
static struct tracker_list *starts_unlocked;

struct locks_on_return {
	int line;
	struct tracker_list *locked;
	struct tracker_list *unlocked;
	struct tracker_list *impossible;
	struct range_list *return_values;
};
DECLARE_PTR_LIST(return_list, struct locks_on_return);
static struct return_list *all_returns;

static char *make_full_name(const char *lock, const char *var)
{
	static char tmp_buf[512];

	snprintf(tmp_buf, sizeof(tmp_buf), "%s:%s", lock, var);
	remove_parens(tmp_buf);
	return alloc_string(tmp_buf);
}

static struct expression *remove_spinlock_check(struct expression *expr)
{
	if (expr->type != EXPR_CALL)
		return expr;
	if (expr->fn->type != EXPR_SYMBOL)
		return expr;
	if (strcmp(expr->fn->symbol_name->name, "spinlock_check"))
		return expr;
	expr = get_argument_from_call_expr(expr->args, 0);
	return expr;
}

static char *get_full_name(struct expression *expr, int index)
{
	struct expression *arg;
	char *name = NULL;
	char *full_name = NULL;
	struct lock_info *lock = &lock_table[index];

	if (lock->arg == RETURN_VAL) {
		name = expr_to_var(expr->left);
		full_name = make_full_name(lock->name, name);
	} else if (lock->arg == NO_ARG) {
		full_name = make_full_name(lock->name, "");
	} else {
		arg = get_argument_from_call_expr(expr->args, lock->arg);
		if (!arg)
			goto free;
		arg = remove_spinlock_check(arg);
		name = expr_to_str(arg);
		if (!name)
			goto free;
		full_name = make_full_name(lock->name, name);
	}
free:
	free_string(name);
	return full_name;
}

static struct smatch_state *get_start_state(struct sm_state *sm)
{
	int is_locked = 0;
	int is_unlocked = 0;

	if (in_tracker_list(starts_locked, my_id, sm->name, sm->sym))
		is_locked = 1;
	if (in_tracker_list(starts_unlocked, my_id, sm->name, sm->sym))
		is_unlocked = 1;
	if (is_locked && is_unlocked)
		return &undefined;
	if (is_locked)
		return &locked;
	if (is_unlocked)
		return &unlocked;
	return &undefined;
}

static struct smatch_state *unmatched_state(struct sm_state *sm)
{
	return &start_state;
}

static void pre_merge_hook(struct sm_state *cur, struct sm_state *other)
{
	if (is_impossible_path())
		set_state(my_id, cur->name, cur->sym, &impossible);
}

static bool nestable(const char *name)
{
	if (strstr(name, "read_sem:"))
		return true;
	if (strcmp(name, "bottom_half:") == 0)
		return true;
	return false;
}

static void do_lock(const char *name)
{
	struct sm_state *sm;

	if (__inline_fn)
		return;

	sm = get_sm_state(my_id, name, NULL);
	if (!sm)
		add_tracker(&starts_unlocked, my_id, name, NULL);
	if (sm && slist_has_state(sm->possible, &locked) && !nestable(name))
		sm_error("double lock '%s'", name);
	if (sm)
		func_has_transition = TRUE;
	set_state(my_id, name, NULL, &locked);
}

static void do_lock_failed(const char *name)
{
	struct sm_state *sm;

	if (__inline_fn)
		return;

	sm = get_sm_state(my_id, name, NULL);
	if (!sm)
		add_tracker(&starts_unlocked, my_id, name, NULL);
	set_state(my_id, name, NULL, &unlocked);
}

static void do_unlock(const char *name)
{
	struct sm_state *sm;

	if (__inline_fn)
		return;
	if (__path_is_null())
		return;
	sm = get_sm_state(my_id, name, NULL);
	if (!sm)
		add_tracker(&starts_locked, my_id, name, NULL);
	if (sm && slist_has_state(sm->possible, &unlocked) &&
			strcmp(name, "bottom_half:") != 0)
		sm_error("double unlock '%s'", name);
	if (sm)
		func_has_transition = TRUE;
	set_state(my_id, name, NULL, &unlocked);
}

static void match_lock_held(const char *fn, struct expression *call_expr,
			struct expression *assign_expr, void *_index)
{
	int index = PTR_INT(_index);
	char *lock_name;
	struct lock_info *lock = &lock_table[index];

	if (lock->arg == NO_ARG) {
		lock_name = get_full_name(NULL, index);
	} else if (lock->arg == RETURN_VAL) {
		if (!assign_expr)
			return;
		lock_name = get_full_name(assign_expr, index);
	} else {
		lock_name = get_full_name(call_expr, index);
	}
	if (!lock_name)
		return;
	do_lock(lock_name);
	free_string(lock_name);
}

static void match_lock_failed(const char *fn, struct expression *call_expr,
			struct expression *assign_expr, void *_index)
{
	int index = PTR_INT(_index);
	char *lock_name;
	struct lock_info *lock = &lock_table[index];

	if (lock->arg == NO_ARG) {
		lock_name = get_full_name(NULL, index);
	} else if (lock->arg == RETURN_VAL) {
		if (!assign_expr)
			return;
		lock_name = get_full_name(assign_expr, index);
	} else {
		lock_name = get_full_name(call_expr, index);
	}
	if (!lock_name)
		return;
	do_lock_failed(lock_name);
	free_string(lock_name);
}

static void match_returns_locked(const char *fn, struct expression *expr,
				      void *_index)
{
	char *full_name = NULL;
	int index = PTR_INT(_index);
	struct lock_info *lock = &lock_table[index];

	if (lock->arg != RETURN_VAL)
		return;
	full_name = get_full_name(expr, index);
	do_lock(full_name);
}

static void match_lock_unlock(const char *fn, struct expression *expr, void *_index)
{
	char *full_name = NULL;
	int index = PTR_INT(_index);
	struct lock_info *lock = &lock_table[index];

	if (__inline_fn)
		return;

	full_name = get_full_name(expr, index);
	if (!full_name)
		return;
	if (lock->action == LOCK)
		do_lock(full_name);
	else
		do_unlock(full_name);
	free_string(full_name);
}

static struct locks_on_return *alloc_return(struct expression *expr)
{
	struct locks_on_return *ret;

	ret = malloc(sizeof(*ret));
	if (!get_implied_rl(expr, &ret->return_values))
		ret->return_values = NULL;
	ret->line = get_lineno();
	ret->locked = NULL;
	ret->unlocked = NULL;
	ret->impossible = NULL;
	return ret;
}

static int check_possible(struct sm_state *sm)
{
	struct sm_state *tmp;
	int islocked = 0;
	int isunlocked = 0;
	int undef = 0;

	if (!option_spammy)
		return 0;

	FOR_EACH_PTR(sm->possible, tmp) {
		if (tmp->state == &locked)
			islocked = 1;
		if (tmp->state == &unlocked)
			isunlocked = 1;
		if (tmp->state == &start_state) {
			struct smatch_state *s;

			s = get_start_state(tmp);
			if (s == &locked)
				islocked = 1;
			else if (s == &unlocked)
				isunlocked = 1;
			else
				undef = 1;
		}
		if (tmp->state == &undefined)
			undef = 1;  // i don't think this is possible any more.
	} END_FOR_EACH_PTR(tmp);
	if ((islocked && isunlocked) || undef) {
		sm_warning("'%s' is sometimes locked here and sometimes unlocked.", sm->name);
		return 1;
	}
	return 0;
}

static struct position warned_pos;

static void match_return(int return_id, char *return_ranges, struct expression *expr)
{
	struct locks_on_return *ret;
	struct stree *stree;
	struct sm_state *tmp;

	if (!final_pass)
		return;
	if (__inline_fn)
		return;

	if (expr && cmp_pos(expr->pos, warned_pos) == 0)
		return;

	ret = alloc_return(expr);

	stree = __get_cur_stree();
	FOR_EACH_MY_SM(my_id, stree, tmp) {
		if (tmp->state == &locked) {
			add_tracker(&ret->locked, tmp->owner, tmp->name,
				tmp->sym);
		} else if (tmp->state == &unlocked) {
			add_tracker(&ret->unlocked, tmp->owner, tmp->name,
				tmp->sym);
		} else if (tmp->state == &start_state) {
			struct smatch_state *s;

			s = get_start_state(tmp);
			if (s == &locked)
				add_tracker(&ret->locked, tmp->owner, tmp->name, 
					    tmp->sym);
			if (s == &unlocked)
				add_tracker(&ret->unlocked, tmp->owner,tmp->name,
					     tmp->sym);
		} else if (tmp->state == &impossible) {
			add_tracker(&ret->impossible, tmp->owner, tmp->name,
				    tmp->sym);
		} else {
			if (check_possible(tmp)) {
				if (expr)
					warned_pos = expr->pos;
			}
		}
	} END_FOR_EACH_SM(tmp);
	add_ptr_list(&all_returns, ret);
}

static void add_line(struct range_list **rl, int line)
{
	sval_t sval = sval_type_val(&int_ctype, line);

	add_range(rl, sval, sval);
}

static int line_printed(struct range_list *rl, int line)
{
	sval_t sval = sval_type_val(&int_ctype, line);

	return rl_has_sval(rl, sval);
}

static void print_inconsistent_returns(struct tracker *lock,
				struct smatch_state *start)
{
	struct locks_on_return *tmp;
	struct range_list *printed = NULL;
	int i;

	sm_warning("inconsistent returns '%s'.", lock->name);
	sm_printf("  Locked on:   ");

	i = 0;
	FOR_EACH_PTR(all_returns, tmp) {
		if (line_printed(printed, tmp->line))
			continue;
		if (in_tracker_list(tmp->unlocked, lock->owner, lock->name, lock->sym))
			continue;
		if (in_tracker_list(tmp->locked, lock->owner, lock->name, lock->sym)) {
			if (i++)
				sm_printf("               ");
			sm_printf("line %d\n", tmp->line);
			add_line(&printed, tmp->line);
			continue;
		}
		if (start == &locked) {
			if (i++)
				sm_printf("               ");
			sm_printf("line %d\n", tmp->line);
			add_line(&printed, tmp->line);
		}
	} END_FOR_EACH_PTR(tmp);

	sm_printf("  Unlocked on: ");
	printed = NULL;
	i = 0;
	FOR_EACH_PTR(all_returns, tmp) {
		if (line_printed(printed, tmp->line))
			continue;
		if (in_tracker_list(tmp->unlocked, lock->owner, lock->name, lock->sym)) {
			if (i++)
				sm_printf("               ");
			sm_printf("line %d\n", tmp->line);
			add_line(&printed, tmp->line);
			continue;
		}
		if (in_tracker_list(tmp->locked, lock->owner, lock->name, lock->sym))
			continue;
		if (start == &unlocked) {
			if (i++)
				sm_printf("               ");
			sm_printf("line %d\n", tmp->line);
			add_line(&printed, tmp->line);
		}
	} END_FOR_EACH_PTR(tmp);
}

static int matches_return_type(struct range_list *rl, enum return_type type)
{
	sval_t zero_sval = ll_to_sval(0);
	sval_t one_sval = ll_to_sval(1);

	/* All these double negatives are super ugly!  */

	switch (type) {
	case ret_zero:
		return !possibly_true_rl(rl, SPECIAL_NOTEQUAL, alloc_rl(zero_sval, zero_sval));
	case ret_one:
		return !possibly_true_rl(rl, SPECIAL_NOTEQUAL, alloc_rl(one_sval, one_sval));
	case ret_non_zero:
		return !possibly_true_rl(rl, SPECIAL_EQUAL, alloc_rl(zero_sval, zero_sval));
	case ret_negative:
		return !possibly_true_rl(rl, SPECIAL_GTE, alloc_rl(zero_sval, zero_sval));
	case ret_positive:
		return !possibly_true_rl(rl, '<', alloc_rl(zero_sval, zero_sval));
	case ret_any:
	default:
		return 1;
	}
}

static int match_held(struct tracker *lock, struct locks_on_return *this_return, struct smatch_state *start)
{
	if (in_tracker_list(this_return->impossible, lock->owner, lock->name, lock->sym))
		return 0;
	if (in_tracker_list(this_return->unlocked, lock->owner, lock->name, lock->sym))
		return 0;
	if (in_tracker_list(this_return->locked, lock->owner, lock->name, lock->sym))
		return 1;
	if (start == &unlocked)
		return 0;
	return 1;
}

static int match_released(struct tracker *lock, struct locks_on_return *this_return, struct smatch_state *start)
{
	if (in_tracker_list(this_return->impossible, lock->owner, lock->name, lock->sym))
		return 0;
	if (in_tracker_list(this_return->unlocked, lock->owner, lock->name, lock->sym))
		return 1;
	if (in_tracker_list(this_return->locked, lock->owner, lock->name, lock->sym))
		return 0;
	if (start == &unlocked)
		return 1;
	return 0;
}

static int held_on_return(struct tracker *lock, struct smatch_state *start, enum return_type type)
{
	struct locks_on_return *tmp;

	FOR_EACH_PTR(all_returns, tmp) {
		if (!matches_return_type(tmp->return_values, type))
			continue;
		if (match_held(lock, tmp, start))
			return 1;
	} END_FOR_EACH_PTR(tmp);
	return 0;
}

static int released_on_return(struct tracker *lock, struct smatch_state *start, enum return_type type)
{
	struct locks_on_return *tmp;

	FOR_EACH_PTR(all_returns, tmp) {
		if (!matches_return_type(tmp->return_values, type))
			continue;
		if (match_released(lock, tmp, start))
			return 1;
	} END_FOR_EACH_PTR(tmp);
	return 0;
}

static void check_returns_consistently(struct tracker *lock,
				struct smatch_state *start)
{
	struct symbol *type;

	if (!held_on_return(lock, start, ret_any) ||
	    !released_on_return(lock, start, ret_any))
		return;

	if (held_on_return(lock, start, ret_zero) &&
	    !held_on_return(lock, start, ret_non_zero))
		return;

	if (held_on_return(lock, start, ret_positive) &&
	    !held_on_return(lock, start, ret_zero))
		return;

	if (held_on_return(lock, start, ret_positive) &&
	    !held_on_return(lock, start, ret_negative))
		return;

	type = cur_func_return_type();
	if (type && type->type == SYM_PTR) {
		if (held_on_return(lock, start, ret_non_zero) &&
		    !held_on_return(lock, start, ret_zero))
			return;
	}

	print_inconsistent_returns(lock, start);
}

static void check_consistency(struct symbol *sym)
{
	struct tracker *tmp;

	FOR_EACH_PTR(starts_locked, tmp) {
		if (in_tracker_list(starts_unlocked, tmp->owner, tmp->name,
					tmp->sym))
			sm_error("locking inconsistency.  We assume "
				   "'%s' is both locked and unlocked at the "
				   "start.",
				tmp->name);
	} END_FOR_EACH_PTR(tmp);

	FOR_EACH_PTR(starts_locked, tmp) {
		check_returns_consistently(tmp, &locked);
	} END_FOR_EACH_PTR(tmp);

	FOR_EACH_PTR(starts_unlocked, tmp) {
		check_returns_consistently(tmp, &unlocked);
	} END_FOR_EACH_PTR(tmp);
}

static void clear_lists(void)
{
	struct locks_on_return *tmp;

	func_has_transition = FALSE;

	free_trackers_and_list(&starts_locked);
	free_trackers_and_list(&starts_unlocked);

	FOR_EACH_PTR(all_returns, tmp) {
		free_trackers_and_list(&tmp->locked);
		free_trackers_and_list(&tmp->unlocked);
		free(tmp);
	} END_FOR_EACH_PTR(tmp);
	__free_ptr_list((struct ptr_list **)&all_returns);
}

static void match_func_end(struct symbol *sym)
{
	if (__inline_fn)
		return;

	if (func_has_transition)
		check_consistency(sym);
}

static void match_after_func(struct symbol *sym)
{
	if (__inline_fn)
		return;
	clear_lists();
}

static void register_lock(int index)
{
	struct lock_info *lock = &lock_table[index];
	void *idx = INT_PTR(index);

	if (lock->return_type == ret_non_zero) {
		return_implies_state(lock->function, 1, INT_MAX, &match_lock_held, idx);
		return_implies_state(lock->function, 0, 0, &match_lock_failed, idx);
	} else if (lock->return_type == ret_any && lock->arg == RETURN_VAL) {
		add_function_assign_hook(lock->function, &match_returns_locked, idx);
	} else if (lock->return_type == ret_any) {
		add_function_hook(lock->function, &match_lock_unlock, idx);
	} else if (lock->return_type == ret_zero) {
		return_implies_state(lock->function, 0, 0, &match_lock_held, idx);
		return_implies_state(lock->function, -4095, -1, &match_lock_failed, idx);
	} else if (lock->return_type == ret_one) {
		return_implies_state(lock->function, 1, 1, &match_lock_held, idx);
		return_implies_state(lock->function, 0, 0, &match_lock_failed, idx);
	}
}

static void load_table(struct lock_info *_lock_table, int size)
{
	int i;

	lock_table = _lock_table;

	for (i = 0; i < size; i++) {
		if (lock_table[i].action == LOCK)
			register_lock(i);
		else
			add_function_hook(lock_table[i].function, &match_lock_unlock, INT_PTR(i));
	}
}

/* print_held_locks() is used in check_call_tree.c */
void print_held_locks(void)
{
	struct stree *stree;
	struct sm_state *sm;
	int i = 0;

	stree = __get_cur_stree();
	FOR_EACH_MY_SM(my_id, stree, sm) {
		if (sm->state != &locked)
			continue;
		if (i++)
			sm_printf(" ");
		sm_printf("'%s'", sm->name);
	} END_FOR_EACH_SM(sm);
}

void check_locking(int id)
{
	my_id = id;

	if (option_project == PROJ_WINE)
		load_table(wine_lock_table, ARRAY_SIZE(wine_lock_table));
	else if (option_project == PROJ_KERNEL)
		load_table(kernel_lock_table, ARRAY_SIZE(kernel_lock_table));
	else
		return;

	add_unmatched_state_hook(my_id, &unmatched_state);
	add_pre_merge_hook(my_id, &pre_merge_hook);
	add_split_return_callback(match_return);
	add_hook(&match_func_end, END_FUNC_HOOK);
	add_hook(&match_after_func, AFTER_FUNC_HOOK);

}