51#if LWIP_IPV4 && LWIP_UDP
62#define NETBIOS_NAME_LEN 16
66#define NETBIOS_NAME_TTL 300000u
69#define NETB_HFLAG_RESPONSE 0x8000U
70#define NETB_HFLAG_OPCODE 0x7800U
71#define NETB_HFLAG_OPCODE_NAME_QUERY 0x0000U
72#define NETB_HFLAG_AUTHORATIVE 0x0400U
73#define NETB_HFLAG_TRUNCATED 0x0200U
74#define NETB_HFLAG_RECURS_DESIRED 0x0100U
75#define NETB_HFLAG_RECURS_AVAILABLE 0x0080U
76#define NETB_HFLAG_BROADCAST 0x0010U
77#define NETB_HFLAG_REPLYCODE 0x0008U
78#define NETB_HFLAG_REPLYCODE_NOERROR 0x0000U
81#define NETB_QTYPE_NB 0x0020U
82#define NETB_QTYPE_NBSTAT 0x0021U
85#define NETB_NFLAG_UNIQUE 0x8000U
86#define NETB_NFLAG_NODETYPE 0x6000U
87#define NETB_NFLAG_NODETYPE_HNODE 0x6000U
88#define NETB_NFLAG_NODETYPE_MNODE 0x4000U
89#define NETB_NFLAG_NODETYPE_PNODE 0x2000U
90#define NETB_NFLAG_NODETYPE_BNODE 0x0000U
92#define NETB_NFLAG_NAME_IN_CONFLICT 0x0800U
93#define NETB_NFLAG_NAME_IS_ACTIVE 0x0400U
94#define NETB_NFLAG_NAME_IS_PERMANENT 0x0200U
97#ifdef PACK_STRUCT_USE_INCLUDES
110#ifdef PACK_STRUCT_USE_INCLUDES
115#ifdef PACK_STRUCT_USE_INCLUDES
119struct netbios_question_hdr {
126#ifdef PACK_STRUCT_USE_INCLUDES
131#ifdef PACK_STRUCT_USE_INCLUDES
135struct netbios_name_hdr {
146#ifdef PACK_STRUCT_USE_INCLUDES
151#ifdef PACK_STRUCT_USE_INCLUDES
156 struct netbios_hdr resp_hdr;
157 struct netbios_name_hdr resp_name;
160#ifdef PACK_STRUCT_USE_INCLUDES
165#ifdef PACK_STRUCT_USE_INCLUDES
169struct netbios_answer {
170 struct netbios_hdr answer_hdr;
179#define OFFSETOF_STRUCT_NETBIOS_ANSWER_NUMBER_OF_NAMES 56
228#ifdef PACK_STRUCT_USE_INCLUDES
232#ifdef NETBIOS_LWIP_NAME
233#define NETBIOS_LOCAL_NAME NETBIOS_LWIP_NAME
236#define NETBIOS_LOCAL_NAME netbiosns_local_name
239static struct udp_pcb *netbiosns_pcb;
243netbiosns_name_decode(
const char *name_enc,
char *name_dec,
int name_dec_len)
269 cnbname = cname << 4;
284 name_dec[
idx++] = (cnbname !=
' ' ? cnbname :
'\0');
295netbiosns_name_encode(
char *name_enc,
char *name_dec,
int name_dec_len)
299 unsigned char ucname;
315 if ((cname <
'A' || cname >
'Z') && (cname <
'0' || cname >
'9')) {
321 if (
idx >= name_dec_len) {
327 name_dec[
idx++] = (
'A' + ((ucname >> 4) & 0x0F));
328 name_dec[
idx++] = (
'A' + ( ucname & 0x0F));
333 for (;
idx < name_dec_len - 1;) {
334 name_dec[
idx++] =
'C';
335 name_dec[
idx++] =
'A';
339 name_dec[
idx] =
'\0';
354 struct netbios_hdr *netbios_hdr = (
struct netbios_hdr *)
p->payload;
355 struct netbios_question_hdr *netbios_question_hdr = (
struct netbios_question_hdr *)(netbios_hdr + 1);
358 if (
p->len < (
sizeof(
struct netbios_hdr) +
sizeof(
struct netbios_question_hdr))) {
367 if (((netbios_hdr->flags &
PP_NTOHS(NETB_HFLAG_OPCODE)) ==
PP_NTOHS(NETB_HFLAG_OPCODE_NAME_QUERY)) &&
368 ((netbios_hdr->flags &
PP_NTOHS(NETB_HFLAG_RESPONSE)) == 0) &&
369 (netbios_hdr->questions ==
PP_NTOHS(1))) {
371 netbiosns_name_decode((
char *)(netbios_question_hdr->encname), netbios_name,
sizeof(netbios_name));
373 if (netbios_question_hdr->type ==
PP_HTONS(NETB_QTYPE_NB)) {
375 if (
lwip_strnicmp(netbios_name, NETBIOS_LOCAL_NAME,
sizeof(NETBIOS_LOCAL_NAME)) == 0) {
377 struct netbios_resp *resp;
381 resp = (
struct netbios_resp *)
q->payload;
384 resp->resp_hdr.trans_id = netbios_hdr->trans_id;
385 resp->resp_hdr.flags =
PP_HTONS(NETB_HFLAG_RESPONSE |
386 NETB_HFLAG_OPCODE_NAME_QUERY |
387 NETB_HFLAG_AUTHORATIVE |
388 NETB_HFLAG_RECURS_DESIRED);
389 resp->resp_hdr.questions = 0;
390 resp->resp_hdr.answerRRs =
PP_HTONS(1);
391 resp->resp_hdr.authorityRRs = 0;
392 resp->resp_hdr.additionalRRs = 0;
395 MEMCPY( resp->resp_name.encname, netbios_question_hdr->encname,
sizeof(netbios_question_hdr->encname));
396 resp->resp_name.nametype = netbios_question_hdr->nametype;
397 resp->resp_name.type = netbios_question_hdr->type;
398 resp->resp_name.cls = netbios_question_hdr->cls;
399 resp->resp_name.ttl =
PP_HTONL(NETBIOS_NAME_TTL);
400 resp->resp_name.datalen =
PP_HTONS(
sizeof(resp->resp_name.flags) +
sizeof(resp->resp_name.addr));
401 resp->resp_name.flags =
PP_HTONS(NETB_NFLAG_NODETYPE_BNODE);
402 ip4_addr_copy(resp->resp_name.addr, *netif_ip4_addr(
netif_default));
411#if LWIP_NETBIOS_RESPOND_NAME_QUERY
412 }
else if (netbios_question_hdr->type ==
PP_HTONS(NETB_QTYPE_NBSTAT)) {
414 if (!
lwip_strnicmp(netbios_name, NETBIOS_LOCAL_NAME,
sizeof(NETBIOS_LOCAL_NAME)) ||
415 !
lwip_strnicmp(netbios_name,
"*",
sizeof(NETBIOS_LOCAL_NAME))) {
418 struct netbios_answer *resp;
423 resp = (
struct netbios_answer *)
q->payload;
426 memset(resp, 0,
sizeof(*resp));
429 resp->answer_hdr.trans_id = netbios_hdr->trans_id;
431 resp->answer_hdr.flags =
PP_HTONS(NETB_HFLAG_RESPONSE | NETB_HFLAG_OPCODE_NAME_QUERY | NETB_HFLAG_AUTHORATIVE);
434 resp->answer_hdr.answerRRs =
PP_HTONS(1);
438 resp->name_size = netbios_question_hdr->nametype;
446 resp->data_length =
PP_HTONS(
sizeof(
struct netbios_answer) -
offsetof(
struct netbios_answer, number_of_names));
447 resp->number_of_names = 1;
452 MEMCPY(resp->answer_name, NETBIOS_LOCAL_NAME,
strlen(NETBIOS_LOCAL_NAME));
455 resp->answer_name_flags =
PP_HTONS(NETB_NFLAG_NAME_IS_ACTIVE);
481#ifdef NETBIOS_LWIP_NAME
486 if (netbiosns_pcb !=
NULL) {
490 udp_recv(netbiosns_pcb, netbiosns_recv, netbiosns_pcb);
494#ifndef NETBIOS_LWIP_NAME
512 for (
i = 0;
i < copy_len;
i++ ) {
515 netbiosns_local_name[copy_len] =
'\0';
527 if (netbiosns_pcb !=
NULL) {
528 udp_remove(netbiosns_pcb);
529 netbiosns_pcb =
NULL;
ACPI_SIZE strlen(const char *String)
#define PACK_STRUCT_STRUCT
#define LWIP_ASSERT(message, assertion)
#define ip_set_option(pcb, opt)
#define ip_current_input_netif()
GLuint GLuint GLsizei GLenum type
GLdouble GLdouble GLdouble GLdouble q
GLenum const GLvoid * addr
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
#define LWIP_UNUSED_ARG(x)
#define PACK_STRUCT_BEGIN
#define PACK_STRUCT_FLD_8(x)
#define PACK_STRUCT_FIELD(x)
#define PACK_STRUCT_FLD_S(x)
#define LWIP_ASSERT_CORE_LOCKED()
#define SMEMCPY(dst, src, len)
struct pbuf * pbuf_alloc(pbuf_layer layer, u16_t length, pbuf_type type)
u8_t pbuf_free(struct pbuf *p)
int lwip_strnicmp(const char *str1, const char *str2, size_t len)
int const JOCTET unsigned int datalen
#define MEMCPY(DST, SRC, BYTES)
void netbiosns_init(void)
void netbiosns_set_name(const char *hostname)
void netbiosns_stop(void)
struct netif * netif_default
typedefPACK_STRUCT_END struct ip4_addr_packed ip4_addr_p_t
#define offsetof(TYPE, MEMBER)