capiutil.c

来自「linux 内核源代码」· C语言 代码 · 共 1,015 行 · 第 1/2 页

C
1,015
字号
	return 0;}/*-------------------------------------------------------*/static void message_2_pars(_cmsg * cmsg){	for (; TYP != _CEND; cmsg->p++) {		switch (TYP) {		case _CBYTE:			byteTRcpy(cmsg->m + cmsg->l, OFF);			cmsg->l++;			break;		case _CWORD:			wordTRcpy(cmsg->m + cmsg->l, OFF);			cmsg->l += 2;			break;		case _CDWORD:			dwordTRcpy(cmsg->m + cmsg->l, OFF);			cmsg->l += 4;			break;		case _CSTRUCT:			*(u8 **) OFF = cmsg->m + cmsg->l;			if (cmsg->m[cmsg->l] != 0xff)				cmsg->l += 1 + cmsg->m[cmsg->l];			else				cmsg->l += 3 + *(u16 *) (cmsg->m + cmsg->l + 1);			break;		case _CMSTRUCT:/*----- Metastruktur 0 -----*/			if (cmsg->m[cmsg->l] == '\0') {				*(_cmstruct *) OFF = CAPI_DEFAULT;				cmsg->l++;				jumpcstruct(cmsg);			} else {				unsigned _l = cmsg->l;				*(_cmstruct *) OFF = CAPI_COMPOSE;				cmsg->l = (cmsg->m + _l)[0] == 255 ? cmsg->l + 3 : cmsg->l + 1;				cmsg->p++;				message_2_pars(cmsg);			}			break;		}	}}/*-------------------------------------------------------*/unsigned capi_message2cmsg(_cmsg * cmsg, u8 * msg){	memset(cmsg, 0, sizeof(_cmsg));	cmsg->m = msg;	cmsg->l = 8;	cmsg->p = 0;	byteTRcpy(cmsg->m + 4, &cmsg->Command);	byteTRcpy(cmsg->m + 5, &cmsg->Subcommand);	cmsg->par = cpars[command_2_index(cmsg->Command, cmsg->Subcommand)];	message_2_pars(cmsg);	wordTRcpy(msg + 0, &cmsg->l);	wordTRcpy(cmsg->m + 2, &cmsg->ApplId);	wordTRcpy(cmsg->m + 6, &cmsg->Messagenumber);	return 0;}/*-------------------------------------------------------*/unsigned capi_cmsg_header(_cmsg * cmsg, u16 _ApplId,			  u8 _Command, u8 _Subcommand,			  u16 _Messagenumber, u32 _Controller){	memset(cmsg, 0, sizeof(_cmsg));	cmsg->ApplId = _ApplId;	cmsg->Command = _Command;	cmsg->Subcommand = _Subcommand;	cmsg->Messagenumber = _Messagenumber;	cmsg->adr.adrController = _Controller;	return 0;}/*-------------------------------------------------------*/static char *mnames[] ={	[0x01] = "ALERT_REQ",	[0x02] = "CONNECT_REQ",	[0x04] = "DISCONNECT_REQ",	[0x05] = "LISTEN_REQ",	[0x08] = "INFO_REQ",	[0x09] = "FACILITY_REQ",	[0x0a] = "SELECT_B_PROTOCOL_REQ",	[0x0b] = "CONNECT_B3_REQ",	[0x0d] = "DISCONNECT_B3_REQ",	[0x0f] = "DATA_B3_REQ",	[0x10] = "RESET_B3_REQ",	[0x13] = "ALERT_CONF",	[0x14] = "CONNECT_CONF",	[0x16] = "DISCONNECT_CONF",	[0x17] = "LISTEN_CONF",	[0x18] = "MANUFACTURER_REQ",	[0x1a] = "INFO_CONF",	[0x1b] = "FACILITY_CONF",	[0x1c] = "SELECT_B_PROTOCOL_CONF",	[0x1d] = "CONNECT_B3_CONF",	[0x1f] = "DISCONNECT_B3_CONF",	[0x21] = "DATA_B3_CONF",	[0x22] = "RESET_B3_CONF",	[0x26] = "CONNECT_IND",	[0x27] = "CONNECT_ACTIVE_IND",	[0x28] = "DISCONNECT_IND",	[0x2a] = "MANUFACTURER_CONF",	[0x2c] = "INFO_IND",	[0x2d] = "FACILITY_IND",	[0x2f] = "CONNECT_B3_IND",	[0x30] = "CONNECT_B3_ACTIVE_IND",	[0x31] = "DISCONNECT_B3_IND",	[0x33] = "DATA_B3_IND",	[0x34] = "RESET_B3_IND",	[0x35] = "CONNECT_B3_T90_ACTIVE_IND",	[0x38] = "CONNECT_RESP",	[0x39] = "CONNECT_ACTIVE_RESP",	[0x3a] = "DISCONNECT_RESP",	[0x3c] = "MANUFACTURER_IND",	[0x3e] = "INFO_RESP",	[0x3f] = "FACILITY_RESP",	[0x41] = "CONNECT_B3_RESP",	[0x42] = "CONNECT_B3_ACTIVE_RESP",	[0x43] = "DISCONNECT_B3_RESP",	[0x45] = "DATA_B3_RESP",	[0x46] = "RESET_B3_RESP",	[0x47] = "CONNECT_B3_T90_ACTIVE_RESP",	[0x4e] = "MANUFACTURER_RESP"};char *capi_cmd2str(u8 cmd, u8 subcmd){	return mnames[command_2_index(cmd, subcmd)];}/*-------------------------------------------------------*/#ifdef CONFIG_CAPI_TRACE/*-------------------------------------------------------*/static char *pnames[] ={    /*00 */ NULL,    /*01 */ NULL,    /*02 */ NULL,    /*03 */ "Controller/PLCI/NCCI",    /*04 */ "AdditionalInfo",    /*05 */ "B1configuration",    /*06 */ "B1protocol",    /*07 */ "B2configuration",    /*08 */ "B2protocol",    /*09 */ "B3configuration",    /*0a */ "B3protocol",    /*0b */ "BC",    /*0c */ "BChannelinformation",    /*0d */ "BProtocol",    /*0e */ "CalledPartyNumber",    /*0f */ "CalledPartySubaddress",    /*10 */ "CallingPartyNumber",    /*11 */ "CallingPartySubaddress",    /*12 */ "CIPmask",    /*13 */ "CIPmask2",    /*14 */ "CIPValue",    /*15 */ "Class",    /*16 */ "ConnectedNumber",    /*17 */ "ConnectedSubaddress",    /*18 */ "Data32",    /*19 */ "DataHandle",    /*1a */ "DataLength",    /*1b */ "FacilityConfirmationParameter",    /*1c */ "Facilitydataarray",    /*1d */ "FacilityIndicationParameter",    /*1e */ "FacilityRequestParameter",    /*1f */ "FacilitySelector",    /*20 */ "Flags",    /*21 */ "Function",    /*22 */ "HLC",    /*23 */ "Info",    /*24 */ "InfoElement",    /*25 */ "InfoMask",    /*26 */ "InfoNumber",    /*27 */ "Keypadfacility",    /*28 */ "LLC",    /*29 */ "ManuData",    /*2a */ "ManuID",    /*2b */ "NCPI",    /*2c */ "Reason",    /*2d */ "Reason_B3",    /*2e */ "Reject",    /*2f */ "Useruserdata"};#include <stdarg.h>/*-------------------------------------------------------*/static _cdebbuf *bufprint(_cdebbuf *cdb, char *fmt,...){	va_list f;	size_t n,r;	if (!cdb)		return NULL;	va_start(f, fmt);	r = cdb->size - cdb->pos;	n = vsnprintf(cdb->p, r, fmt, f);	va_end(f);	if (n >= r) {		/* truncated, need bigger buffer */		size_t ns = 2 * cdb->size;		u_char *nb;		while ((ns - cdb->pos) <= n)			ns *= 2;		nb = kmalloc(ns, GFP_ATOMIC);		if (!nb) {			cdebbuf_free(cdb);			return NULL;		}		memcpy(nb, cdb->buf, cdb->pos);		kfree(cdb->buf);		nb[cdb->pos] = 0;		cdb->buf = nb;		cdb->p = cdb->buf + cdb->pos;		cdb->size = ns;		va_start(f, fmt);		r = cdb->size - cdb->pos;		n = vsnprintf(cdb->p, r, fmt, f);		va_end(f);	}	cdb->p += n;	cdb->pos += n;	return cdb;}static _cdebbuf *printstructlen(_cdebbuf *cdb, u8 * m, unsigned len){	unsigned hex = 0;	if (!cdb)		return NULL;	for (; len; len--, m++)		if (isalnum(*m) || *m == ' ') {			if (hex)				cdb = bufprint(cdb, ">");			cdb = bufprint(cdb, "%c", *m);			hex = 0;		} else {			if (!hex)				cdb = bufprint(cdb, "<%02x", *m);			else				cdb = bufprint(cdb, " %02x", *m);			hex = 1;		}	if (hex)		cdb = bufprint(cdb, ">");	return cdb;}static _cdebbuf *printstruct(_cdebbuf *cdb, u8 * m){	unsigned len;	if (m[0] != 0xff) {		len = m[0];		m += 1;	} else {		len = ((u16 *) (m + 1))[0];		m += 3;	}	cdb = printstructlen(cdb, m, len);	return cdb;}/*-------------------------------------------------------*/#define NAME (pnames[cmsg->par[cmsg->p]])static _cdebbuf *protocol_message_2_pars(_cdebbuf *cdb, _cmsg *cmsg, int level){	for (; TYP != _CEND; cmsg->p++) {		int slen = 29 + 3 - level;		int i;		if (!cdb)			return NULL;		cdb = bufprint(cdb, "  ");		for (i = 0; i < level - 1; i++)			cdb = bufprint(cdb, " ");		switch (TYP) {		case _CBYTE:			cdb = bufprint(cdb, "%-*s = 0x%x\n", slen, NAME, *(u8 *) (cmsg->m + cmsg->l));			cmsg->l++;			break;		case _CWORD:			cdb = bufprint(cdb, "%-*s = 0x%x\n", slen, NAME, *(u16 *) (cmsg->m + cmsg->l));			cmsg->l += 2;			break;		case _CDWORD:			cdb = bufprint(cdb, "%-*s = 0x%lx\n", slen, NAME, *(u32 *) (cmsg->m + cmsg->l));			cmsg->l += 4;			break;		case _CSTRUCT:			cdb = bufprint(cdb, "%-*s = ", slen, NAME);			if (cmsg->m[cmsg->l] == '\0')				cdb = bufprint(cdb, "default");			else				cdb = printstruct(cdb, cmsg->m + cmsg->l);			cdb = bufprint(cdb, "\n");			if (cmsg->m[cmsg->l] != 0xff)				cmsg->l += 1 + cmsg->m[cmsg->l];			else				cmsg->l += 3 + *(u16 *) (cmsg->m + cmsg->l + 1);			break;		case _CMSTRUCT:/*----- Metastruktur 0 -----*/			if (cmsg->m[cmsg->l] == '\0') {				cdb = bufprint(cdb, "%-*s = default\n", slen, NAME);				cmsg->l++;				jumpcstruct(cmsg);			} else {				char *name = NAME;				unsigned _l = cmsg->l;				cdb = bufprint(cdb, "%-*s\n", slen, name);				cmsg->l = (cmsg->m + _l)[0] == 255 ? cmsg->l + 3 : cmsg->l + 1;				cmsg->p++;				cdb = protocol_message_2_pars(cdb, cmsg, level + 1);			}			break;		}	}	return cdb;}/*-------------------------------------------------------*/static _cdebbuf *g_debbuf;static u_long g_debbuf_lock;static _cmsg *g_cmsg;static _cdebbuf *cdebbuf_alloc(void){	_cdebbuf *cdb;	if (likely(!test_and_set_bit(1, &g_debbuf_lock))) {		cdb = g_debbuf;		goto init;	} else		cdb = kmalloc(sizeof(_cdebbuf), GFP_ATOMIC);	if (!cdb)		return NULL;	cdb->buf = kmalloc(CDEBUG_SIZE, GFP_ATOMIC);	if (!cdb->buf) {		kfree(cdb);		return NULL;	}	cdb->size = CDEBUG_SIZE;init:	cdb->buf[0] = 0;	cdb->p = cdb->buf;	cdb->pos = 0;	return cdb;}void cdebbuf_free(_cdebbuf *cdb){	if (likely(cdb == g_debbuf)) {		test_and_clear_bit(1, &g_debbuf_lock);		return;	}	if (likely(cdb))		kfree(cdb->buf);	kfree(cdb);}_cdebbuf *capi_message2str(u8 * msg){	_cdebbuf *cdb;	_cmsg	*cmsg;	cdb = cdebbuf_alloc();	if (unlikely(!cdb))		return NULL;	if (likely(cdb == g_debbuf))		cmsg = g_cmsg;	else		cmsg = kmalloc(sizeof(_cmsg), GFP_ATOMIC);	if (unlikely(!cmsg)) {		cdebbuf_free(cdb);		return NULL;	}	cmsg->m = msg;	cmsg->l = 8;	cmsg->p = 0;	byteTRcpy(cmsg->m + 4, &cmsg->Command);	byteTRcpy(cmsg->m + 5, &cmsg->Subcommand);	cmsg->par = cpars[command_2_index(cmsg->Command, cmsg->Subcommand)];	cdb = bufprint(cdb, "%-26s ID=%03d #0x%04x LEN=%04d\n",		 mnames[command_2_index(cmsg->Command, cmsg->Subcommand)],		 ((unsigned short *) msg)[1],		 ((unsigned short *) msg)[3],		 ((unsigned short *) msg)[0]);	cdb = protocol_message_2_pars(cdb, cmsg, 1);	if (unlikely(cmsg != g_cmsg))		kfree(cmsg);	return cdb;}_cdebbuf *capi_cmsg2str(_cmsg * cmsg){	_cdebbuf *cdb;	cdb = cdebbuf_alloc();	if (!cdb)		return NULL;	cmsg->l = 8;	cmsg->p = 0;	cdb = bufprint(cdb, "%s ID=%03d #0x%04x LEN=%04d\n",		 mnames[command_2_index(cmsg->Command, cmsg->Subcommand)],		 ((u16 *) cmsg->m)[1],		 ((u16 *) cmsg->m)[3],		 ((u16 *) cmsg->m)[0]);	cdb = protocol_message_2_pars(cdb, cmsg, 1);	return cdb;}int __init cdebug_init(void){	g_cmsg= kmalloc(sizeof(_cmsg), GFP_KERNEL);	if (!g_cmsg)		return ENOMEM;	g_debbuf = kmalloc(sizeof(_cdebbuf), GFP_KERNEL);	if (!g_debbuf) {		kfree(g_cmsg);		return ENOMEM;	}	g_debbuf->buf = kmalloc(CDEBUG_GSIZE, GFP_KERNEL);	if (!g_debbuf->buf) {		kfree(g_cmsg);		kfree(g_debbuf);		return ENOMEM;;	}	g_debbuf->size = CDEBUG_GSIZE;	g_debbuf->buf[0] = 0;	g_debbuf->p = g_debbuf->buf;	g_debbuf->pos = 0;	return 0;}void __exit cdebug_exit(void){	if (g_debbuf)		kfree(g_debbuf->buf);	kfree(g_debbuf);	kfree(g_cmsg);}#else /* !CONFIG_CAPI_TRACE */static _cdebbuf g_debbuf = {"CONFIG_CAPI_TRACE not enabled", NULL, 0, 0};_cdebbuf *capi_message2str(u8 * msg){	return &g_debbuf;}_cdebbuf *capi_cmsg2str(_cmsg * cmsg){	return &g_debbuf;}void cdebbuf_free(_cdebbuf *cdb){}int __init cdebug_init(void){	return 0;}void __exit cdebug_exit(void){}#endifEXPORT_SYMBOL(cdebbuf_free);EXPORT_SYMBOL(capi_cmsg2message);EXPORT_SYMBOL(capi_message2cmsg);EXPORT_SYMBOL(capi_cmsg_header);EXPORT_SYMBOL(capi_cmd2str);EXPORT_SYMBOL(capi_cmsg2str);EXPORT_SYMBOL(capi_message2str);EXPORT_SYMBOL(capi_info2str);

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?