Merge commit '1a99df84e12315c63f2e96a2972864e4c311712d'
authorSylvain Munaut <tnt@246tNt.com>
Sat, 30 Jul 2011 19:35:37 +0000 (21:35 +0200)
committerSylvain Munaut <tnt@246tNt.com>
Sat, 30 Jul 2011 19:35:37 +0000 (21:35 +0200)
1  2 
src/shared/libosmocore/include/osmocom/core/logging.h
src/shared/libosmocore/src/gsm/gsm48_ie.c
src/shared/libosmocore/src/logging.c

index 06d90e5,0000000..154ee19
mode 100644,000000..100644
--- /dev/null
@@@ -1,164 -1,0 +1,164 @@@
- #define OSMO_NUM_DLIB 7
 +#ifndef _OSMOCORE_LOGGING_H
 +#define _OSMOCORE_LOGGING_H
 +
 +#include <stdio.h>
 +#include <stdint.h>
 +#include <osmocom/core/linuxlist.h>
 +
 +#define LOG_MAX_CTX           8
 +#define LOG_MAX_FILTERS       8
 +
 +#define DEBUG
 +
 +#ifdef DEBUG
 +#define DEBUGP(ss, fmt, args...) logp(ss, __FILE__, __LINE__, 0, fmt, ## args)
 +#define DEBUGPC(ss, fmt, args...) logp(ss, __FILE__, __LINE__, 1, fmt, ## args)
 +#else
 +#define DEBUGP(xss, fmt, args...)
 +#define DEBUGPC(ss, fmt, args...)
 +#endif
 +
 +
 +void logp(int subsys, char *file, int line, int cont, const char *format, ...) __attribute__ ((format (printf, 5, 6)));
 +
 +/* new logging interface */
 +#define LOGP(ss, level, fmt, args...) \
 +      logp2(ss, level, __FILE__, __LINE__, 0, fmt, ##args)
 +#define LOGPC(ss, level, fmt, args...) \
 +      logp2(ss, level, __FILE__, __LINE__, 1, fmt, ##args)
 +
 +/* different levels */
 +#define LOGL_DEBUG    1       /* debugging information */
 +#define LOGL_INFO     3
 +#define LOGL_NOTICE   5       /* abnormal/unexpected condition */
 +#define LOGL_ERROR    7       /* error condition, requires user action */
 +#define LOGL_FATAL    8       /* fatal, program aborted */
 +
 +#define LOG_FILTER_ALL        0x0001
 +
 +/* logging levels defined by the library itself */
 +#define DLGLOBAL      -1
 +#define DLLAPDM               -2
 +#define DLINP         -3
 +#define DLMUX         -4
 +#define DLMI          -5
 +#define DLMIB         -6
++#define OSMO_NUM_DLIB 6
 +
 +struct log_category {
 +      uint8_t loglevel;
 +      uint8_t enabled;
 +};
 +
 +struct log_info_cat {
 +      const char *name;
 +      const char *color;
 +      const char *description;
 +      uint8_t loglevel;
 +      uint8_t enabled;
 +};
 +
 +/* log context information, passed to filter */
 +struct log_context {
 +      void *ctx[LOG_MAX_CTX+1];
 +};
 +
 +struct log_target;
 +
 +typedef int log_filter(const struct log_context *ctx,
 +                     struct log_target *target);
 +
 +struct log_info {
 +      /* filter callback function */
 +      log_filter *filter_fn;
 +
 +      /* per-category information */
 +      struct log_info_cat *cat;
 +      unsigned int num_cat;
 +      unsigned int num_cat_user;
 +};
 +
 +enum log_target_type {
 +      LOG_TGT_TYPE_VTY,
 +      LOG_TGT_TYPE_SYSLOG,
 +      LOG_TGT_TYPE_FILE,
 +      LOG_TGT_TYPE_STDERR,
 +};
 +
 +struct log_target {
 +        struct llist_head entry;
 +
 +      int filter_map;
 +      void *filter_data[LOG_MAX_FILTERS+1];
 +
 +      struct log_category *categories;
 +
 +      uint8_t loglevel;
 +      unsigned int use_color:1;
 +      unsigned int print_timestamp:1;
 +
 +      enum log_target_type type;
 +
 +      union {
 +              struct {
 +                      FILE *out;
 +                      const char *fname;
 +              } tgt_file;
 +
 +              struct {
 +                      int priority;
 +                      int facility;
 +              } tgt_syslog;
 +
 +              struct {
 +                      void *vty;
 +              } tgt_vty;
 +      };
 +
 +        void (*output) (struct log_target *target, unsigned int level,
 +                      const char *string);
 +};
 +
 +/* use the above macros */
 +void logp2(int subsys, unsigned int level, char *file,
 +         int line, int cont, const char *format, ...)
 +                              __attribute__ ((format (printf, 6, 7)));
 +int log_init(const struct log_info *inf, void *talloc_ctx);
 +
 +/* context management */
 +void log_reset_context(void);
 +int log_set_context(uint8_t ctx, void *value);
 +
 +/* filter on the targets */
 +void log_set_all_filter(struct log_target *target, int);
 +
 +void log_set_use_color(struct log_target *target, int);
 +void log_set_print_timestamp(struct log_target *target, int);
 +void log_set_log_level(struct log_target *target, int log_level);
 +void log_parse_category_mask(struct log_target *target, const char* mask);
 +int log_parse_level(const char *lvl);
 +const char *log_level_str(unsigned int lvl);
 +int log_parse_category(const char *category);
 +void log_set_category_filter(struct log_target *target, int category,
 +                             int enable, int level);
 +
 +/* management of the targets */
 +struct log_target *log_target_create(void);
 +void log_target_destroy(struct log_target *target);
 +struct log_target *log_target_create_stderr(void);
 +struct log_target *log_target_create_file(const char *fname);
 +struct log_target *log_target_create_syslog(const char *ident, int option,
 +                                          int facility);
 +int log_target_file_reopen(struct log_target *tgt);
 +
 +void log_add_target(struct log_target *target);
 +void log_del_target(struct log_target *target);
 +
 +/* Generate command string for VTY use */
 +const char *log_vty_command_string(const struct log_info *info);
 +const char *log_vty_command_description(const struct log_info *info);
 +
 +struct log_target *log_target_find(int type, const char *fname);
 +extern struct llist_head osmo_log_target_list;
 +
 +#endif /* _OSMOCORE_LOGGING_H */
index efcf281,0000000..863e636
mode 100644,000000..100644
--- /dev/null
@@@ -1,1095 -1,0 +1,1095 @@@
-                       f[((w[1]       + ((w[3] - 256 - w[5] - 1) % 511)) % 1023) + 1].mask |= frqt;
 +/* GSM Mobile Radio Interface Layer 3 messages
 + * 3GPP TS 04.08 version 7.21.0 Release 1998 / ETSI TS 100 940 V7.21.0 */
 +
 +/* (C) 2008 by Harald Welte <laforge@gnumonks.org>
 + * (C) 2009-2010 by Andreas Eversberg
 + *
 + * All Rights Reserved
 + *
 + * 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, write to the Free Software Foundation, Inc.,
 + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 + *
 + */
 +
 +
 +#include <stdint.h>
 +#include <string.h>
 +#include <errno.h>
 +
 +#include <osmocom/core/utils.h>
 +#include <osmocom/core/msgb.h>
 +#include <osmocom/gsm/tlv.h>
 +#include <osmocom/gsm/mncc.h>
 +#include <osmocom/gsm/protocol/gsm_04_08.h>
 +#include <osmocom/gsm/gsm48_ie.h>
 +
 +static const char bcd_num_digits[] = {
 +      '0', '1', '2', '3', '4', '5', '6', '7',
 +      '8', '9', '*', '#', 'a', 'b', 'c', '\0'
 +};
 +
 +/* decode a 'called/calling/connect party BCD number' as in 10.5.4.7 */
 +int gsm48_decode_bcd_number(char *output, int output_len,
 +                          const uint8_t *bcd_lv, int h_len)
 +{
 +      uint8_t in_len = bcd_lv[0];
 +      int i;
 +
 +      for (i = 1 + h_len; i <= in_len; i++) {
 +              /* lower nibble */
 +              output_len--;
 +              if (output_len <= 1)
 +                      break;
 +              *output++ = bcd_num_digits[bcd_lv[i] & 0xf];
 +
 +              /* higher nibble */
 +              output_len--;
 +              if (output_len <= 1)
 +                      break;
 +              *output++ = bcd_num_digits[bcd_lv[i] >> 4];
 +      }
 +      if (output_len >= 1)
 +              *output++ = '\0';
 +
 +      return 0;
 +}
 +
 +/* convert a single ASCII character to call-control BCD */
 +static int asc_to_bcd(const char asc)
 +{
 +      int i;
 +
 +      for (i = 0; i < ARRAY_SIZE(bcd_num_digits); i++) {
 +              if (bcd_num_digits[i] == asc)
 +                      return i;
 +      }
 +      return -EINVAL;
 +}
 +
 +/* convert a ASCII phone number to 'called/calling/connect party BCD number' */
 +int gsm48_encode_bcd_number(uint8_t *bcd_lv, uint8_t max_len,
 +                    int h_len, const char *input)
 +{
 +      int in_len = strlen(input);
 +      int i;
 +      uint8_t *bcd_cur = bcd_lv + 1 + h_len;
 +
 +      /* two digits per byte, plus type byte */
 +      bcd_lv[0] = in_len/2 + h_len;
 +      if (in_len % 2)
 +              bcd_lv[0]++;
 +
 +      if (bcd_lv[0] > max_len)
 +              return -EIO;
 +
 +      for (i = 0; i < in_len; i++) {
 +              int rc = asc_to_bcd(input[i]);
 +              if (rc < 0)
 +                      return rc;
 +              if (i % 2 == 0)
 +                      *bcd_cur = rc;
 +              else
 +                      *bcd_cur++ |= (rc << 4);
 +      }
 +      /* append padding nibble in case of odd length */
 +      if (i % 2)
 +              *bcd_cur++ |= 0xf0;
 +
 +      /* return how many bytes we used */
 +      return (bcd_cur - bcd_lv);
 +}
 +
 +/* decode 'bearer capability' */
 +int gsm48_decode_bearer_cap(struct gsm_mncc_bearer_cap *bcap,
 +                           const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +      int i, s;
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      bcap->speech_ver[0] = -1; /* end of list, of maximum 7 values */
 +
 +      /* octet 3 */
 +      bcap->transfer = lv[1] & 0x07;
 +      bcap->mode = (lv[1] & 0x08) >> 3;
 +      bcap->coding = (lv[1] & 0x10) >> 4;
 +      bcap->radio = (lv[1] & 0x60) >> 5;
 +
 +      if (bcap->transfer == GSM_MNCC_BCAP_SPEECH) {
 +              i = 1;
 +              s = 0;
 +              while(!(lv[i] & 0x80)) {
 +                      i++; /* octet 3a etc */
 +                      if (in_len < i)
 +                              return 0;
 +                      bcap->speech_ver[s++] = lv[i] & 0x0f;
 +                      bcap->speech_ver[s] = -1; /* end of list */
 +                      if (i == 2) /* octet 3a */
 +                              bcap->speech_ctm = (lv[i] & 0x20) >> 5;
 +                      if (s == 7) /* maximum speech versions + end of list */
 +                              return 0;
 +              }
 +      } else {
 +              i = 1;
 +              while (!(lv[i] & 0x80)) {
 +                      i++; /* octet 3a etc */
 +                      if (in_len < i)
 +                              return 0;
 +                      /* ignore them */
 +              }
 +              /* FIXME: implement OCTET 4+ parsing */
 +      }
 +
 +      return 0;
 +}
 +
 +/* encode 'bearer capability' */
 +int gsm48_encode_bearer_cap(struct msgb *msg, int lv_only,
 +                           const struct gsm_mncc_bearer_cap *bcap)
 +{
 +      uint8_t lv[32 + 1];
 +      int i = 1, s;
 +
 +      lv[1] = bcap->transfer;
 +      lv[1] |= bcap->mode << 3;
 +      lv[1] |= bcap->coding << 4;
 +      lv[1] |= bcap->radio << 5;
 +
 +      if (bcap->transfer == GSM_MNCC_BCAP_SPEECH) {
 +              for (s = 0; bcap->speech_ver[s] >= 0; s++) {
 +                      i++; /* octet 3a etc */
 +                      lv[i] = bcap->speech_ver[s];
 +                      if (i == 2) /* octet 3a */
 +                              lv[i] |= bcap->speech_ctm << 5;
 +              }
 +              lv[i] |= 0x80; /* last IE of octet 3 etc */
 +      } else {
 +              /* FIXME: implement OCTET 4+ encoding */
 +      }
 +
 +      lv[0] = i;
 +      if (lv_only)
 +              msgb_lv_put(msg, lv[0], lv+1);
 +      else
 +              msgb_tlv_put(msg, GSM48_IE_BEARER_CAP, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'call control cap' */
 +int gsm48_decode_cccap(struct gsm_mncc_cccap *ccap, const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      /* octet 3 */
 +      ccap->dtmf = lv[1] & 0x01;
 +      ccap->pcp = (lv[1] & 0x02) >> 1;
 +
 +      return 0;
 +}
 +
 +/* encode 'call control cap' */
 +int gsm48_encode_cccap(struct msgb *msg,
 +                      const struct gsm_mncc_cccap *ccap)
 +{
 +      uint8_t lv[2];
 +
 +      lv[0] = 1;
 +      lv[1] = 0;
 +      if (ccap->dtmf)
 +              lv [1] |= 0x01;
 +      if (ccap->pcp)
 +              lv [1] |= 0x02;
 +
 +      msgb_tlv_put(msg, GSM48_IE_CC_CAP, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'called party BCD number' */
 +int gsm48_decode_called(struct gsm_mncc_number *called,
 +                       const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      /* octet 3 */
 +      called->plan = lv[1] & 0x0f;
 +      called->type = (lv[1] & 0x70) >> 4;
 +
 +      /* octet 4..N */
 +      gsm48_decode_bcd_number(called->number, sizeof(called->number), lv, 1);
 +
 +      return 0;
 +}
 +
 +/* encode 'called party BCD number' */
 +int gsm48_encode_called(struct msgb *msg,
 +                       const struct gsm_mncc_number *called)
 +{
 +      uint8_t lv[18];
 +      int ret;
 +
 +      /* octet 3 */
 +      lv[1] = 0x80; /* no extension */
 +      lv[1] |= called->plan;
 +      lv[1] |= called->type << 4;
 +
 +      /* octet 4..N, octet 2 */
 +      ret = gsm48_encode_bcd_number(lv, sizeof(lv), 1, called->number);
 +      if (ret < 0)
 +              return ret;
 +
 +      msgb_tlv_put(msg, GSM48_IE_CALLED_BCD, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode callerid of various IEs */
 +int gsm48_decode_callerid(struct gsm_mncc_number *callerid,
 +                       const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +      int i = 1;
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      /* octet 3 */
 +      callerid->plan = lv[1] & 0x0f;
 +      callerid->type = (lv[1] & 0x70) >> 4;
 +
 +      /* octet 3a */
 +      if (!(lv[1] & 0x80)) {
 +              callerid->screen = lv[2] & 0x03;
 +              callerid->present = (lv[2] & 0x60) >> 5;
 +              i = 2;
 +      }
 +
 +      /* octet 4..N */
 +      gsm48_decode_bcd_number(callerid->number, sizeof(callerid->number), lv, i);
 +
 +      return 0;
 +}
 +
 +/* encode callerid of various IEs */
 +int gsm48_encode_callerid(struct msgb *msg, int ie, int max_len,
 +                         const struct gsm_mncc_number *callerid)
 +{
 +      uint8_t lv[max_len - 1];
 +      int h_len = 1;
 +      int ret;
 +
 +      /* octet 3 */
 +      lv[1] = callerid->plan;
 +      lv[1] |= callerid->type << 4;
 +
 +      if (callerid->present || callerid->screen) {
 +              /* octet 3a */
 +              lv[2] = callerid->screen;
 +              lv[2] |= callerid->present << 5;
 +              lv[2] |= 0x80;
 +              h_len++;
 +      } else
 +              lv[1] |= 0x80;
 +
 +      /* octet 4..N, octet 2 */
 +      ret = gsm48_encode_bcd_number(lv, sizeof(lv), h_len, callerid->number);
 +      if (ret < 0)
 +              return ret;
 +
 +      msgb_tlv_put(msg, ie, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'cause' */
 +int gsm48_decode_cause(struct gsm_mncc_cause *cause,
 +                      const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +      int i;
 +
 +      if (in_len < 2)
 +              return -EINVAL;
 +
 +      cause->diag_len = 0;
 +
 +      /* octet 3 */
 +      cause->location = lv[1] & 0x0f;
 +      cause->coding = (lv[1] & 0x60) >> 5;
 +
 +      i = 1;
 +      if (!(lv[i] & 0x80)) {
 +              i++; /* octet 3a */
 +              if (in_len < i+1)
 +                      return 0;
 +              cause->rec = 1;
 +              cause->rec_val = lv[i] & 0x7f;
 +      }
 +      i++;
 +
 +      /* octet 4 */
 +      cause->value = lv[i] & 0x7f;
 +      i++;
 +
 +      if (in_len < i) /* no diag */
 +              return 0;
 +
 +      if (in_len - (i-1) > 32) /* maximum 32 octets */
 +              return 0;
 +
 +      /* octet 5-N */
 +      memcpy(cause->diag, lv + i, in_len - (i-1));
 +      cause->diag_len = in_len - (i-1);
 +
 +      return 0;
 +}
 +
 +/* encode 'cause' */
 +int gsm48_encode_cause(struct msgb *msg, int lv_only,
 +                      const struct gsm_mncc_cause *cause)
 +{
 +      uint8_t lv[32+4];
 +      int i;
 +
 +      if (cause->diag_len > 32)
 +              return -EINVAL;
 +
 +      /* octet 3 */
 +      lv[1] = cause->location;
 +      lv[1] |= cause->coding << 5;
 +
 +      i = 1;
 +      if (cause->rec) {
 +              i++; /* octet 3a */
 +              lv[i] = cause->rec_val;
 +      }
 +      lv[i] |= 0x80; /* end of octet 3 */
 +
 +      /* octet 4 */
 +      i++;
 +      lv[i] = 0x80 | cause->value;
 +
 +      /* octet 5-N */
 +      if (cause->diag_len) {
 +              memcpy(lv + i, cause->diag, cause->diag_len);
 +              i += cause->diag_len;
 +      }
 +
 +      lv[0] = i;
 +      if (lv_only)
 +              msgb_lv_put(msg, lv[0], lv+1);
 +      else
 +              msgb_tlv_put(msg, GSM48_IE_CAUSE, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'calling number' */
 +int gsm48_decode_calling(struct gsm_mncc_number *calling,
 +                       const uint8_t *lv)
 +{
 +      return gsm48_decode_callerid(calling, lv);
 +}
 +
 +/* encode 'calling number' */
 +int gsm48_encode_calling(struct msgb *msg, 
 +                        const struct gsm_mncc_number *calling)
 +{
 +      return gsm48_encode_callerid(msg, GSM48_IE_CALLING_BCD, 14, calling);
 +}
 +
 +/* decode 'connected number' */
 +int gsm48_decode_connected(struct gsm_mncc_number *connected,
 +                       const uint8_t *lv)
 +{
 +      return gsm48_decode_callerid(connected, lv);
 +}
 +
 +/* encode 'connected number' */
 +int gsm48_encode_connected(struct msgb *msg,
 +                          const struct gsm_mncc_number *connected)
 +{
 +      return gsm48_encode_callerid(msg, GSM48_IE_CONN_BCD, 14, connected);
 +}
 +
 +/* decode 'redirecting number' */
 +int gsm48_decode_redirecting(struct gsm_mncc_number *redirecting,
 +                       const uint8_t *lv)
 +{
 +      return gsm48_decode_callerid(redirecting, lv);
 +}
 +
 +/* encode 'redirecting number' */
 +int gsm48_encode_redirecting(struct msgb *msg,
 +                            const struct gsm_mncc_number *redirecting)
 +{
 +      return gsm48_encode_callerid(msg, GSM48_IE_REDIR_BCD, 19, redirecting);
 +}
 +
 +/* decode 'facility' */
 +int gsm48_decode_facility(struct gsm_mncc_facility *facility,
 +                         const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      if (in_len > sizeof(facility->info))
 +              return -EINVAL;
 +
 +      memcpy(facility->info, lv+1, in_len);
 +      facility->len = in_len;
 +
 +      return 0;
 +}
 +
 +/* encode 'facility' */
 +int gsm48_encode_facility(struct msgb *msg, int lv_only,
 +                         const struct gsm_mncc_facility *facility)
 +{
 +      uint8_t lv[GSM_MAX_FACILITY + 1];
 +
 +      if (facility->len < 1 || facility->len > GSM_MAX_FACILITY)
 +              return -EINVAL;
 +
 +      memcpy(lv+1, facility->info, facility->len);
 +      lv[0] = facility->len;
 +      if (lv_only)
 +              msgb_lv_put(msg, lv[0], lv+1);
 +      else
 +              msgb_tlv_put(msg, GSM48_IE_FACILITY, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'notify' */
 +int gsm48_decode_notify(int *notify, const uint8_t *v)
 +{
 +      *notify = v[0] & 0x7f;
 +
 +      return 0;
 +}
 +
 +/* encode 'notify' */
 +int gsm48_encode_notify(struct msgb *msg, int notify)
 +{
 +      msgb_v_put(msg, notify | 0x80);
 +
 +      return 0;
 +}
 +
 +/* decode 'signal' */
 +int gsm48_decode_signal(int *signal, const uint8_t *v)
 +{
 +      *signal = v[0];
 +
 +      return 0;
 +}
 +
 +/* encode 'signal' */
 +int gsm48_encode_signal(struct msgb *msg, int signal)
 +{
 +      msgb_tv_put(msg, GSM48_IE_SIGNAL, signal);
 +
 +      return 0;
 +}
 +
 +/* decode 'keypad' */
 +int gsm48_decode_keypad(int *keypad, const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      *keypad = lv[1] & 0x7f;
 +
 +      return 0;
 +}
 +
 +/* encode 'keypad' */
 +int gsm48_encode_keypad(struct msgb *msg, int keypad)
 +{
 +      msgb_tv_put(msg, GSM48_IE_KPD_FACILITY, keypad);
 +
 +      return 0;
 +}
 +
 +/* decode 'progress' */
 +int gsm48_decode_progress(struct gsm_mncc_progress *progress,
 +                         const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +
 +      if (in_len < 2)
 +              return -EINVAL;
 +
 +      progress->coding = (lv[1] & 0x60) >> 5;
 +      progress->location = lv[1] & 0x0f;
 +      progress->descr = lv[2] & 0x7f;
 +
 +      return 0;
 +}
 +
 +/* encode 'progress' */
 +int gsm48_encode_progress(struct msgb *msg, int lv_only,
 +                         const struct gsm_mncc_progress *p)
 +{
 +      uint8_t lv[3];
 +
 +      lv[0] = 2;
 +      lv[1] = 0x80 | ((p->coding & 0x3) << 5) | (p->location & 0xf);
 +      lv[2] = 0x80 | (p->descr & 0x7f);
 +      if (lv_only)
 +              msgb_lv_put(msg, lv[0], lv+1);
 +      else
 +              msgb_tlv_put(msg, GSM48_IE_PROGR_IND, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'user-user' */
 +int gsm48_decode_useruser(struct gsm_mncc_useruser *uu,
 +                         const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +      char *info = uu->info;
 +      int info_len = sizeof(uu->info);
 +      int i;
 +
 +      if (in_len < 1)
 +              return -EINVAL;
 +
 +      uu->proto = lv[1];
 +
 +      for (i = 2; i <= in_len; i++) {
 +              info_len--;
 +              if (info_len <= 1)
 +                      break;
 +              *info++ = lv[i];
 +      }
 +      if (info_len >= 1)
 +              *info++ = '\0';
 +
 +      return 0;
 +}
 +
 +/* encode 'useruser' */
 +int gsm48_encode_useruser(struct msgb *msg, int lv_only,
 +                         const struct gsm_mncc_useruser *uu)
 +{
 +      uint8_t lv[GSM_MAX_USERUSER + 2];
 +
 +      if (strlen(uu->info) > GSM_MAX_USERUSER)
 +              return -EINVAL;
 +
 +      lv[0] = 1 + strlen(uu->info);
 +      lv[1] = uu->proto;
 +      memcpy(lv + 2, uu->info, strlen(uu->info));
 +      if (lv_only)
 +              msgb_lv_put(msg, lv[0], lv+1);
 +      else
 +              msgb_tlv_put(msg, GSM48_IE_USER_USER, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'ss version' */
 +int gsm48_decode_ssversion(struct gsm_mncc_ssversion *ssv,
 +                          const uint8_t *lv)
 +{
 +      uint8_t in_len = lv[0];
 +
 +      if (in_len < 1 || in_len < sizeof(ssv->info))
 +              return -EINVAL;
 +
 +      memcpy(ssv->info, lv + 1, in_len);
 +      ssv->len = in_len;
 +
 +      return 0;
 +}
 +
 +/* encode 'ss version' */
 +int gsm48_encode_ssversion(struct msgb *msg,
 +                         const struct gsm_mncc_ssversion *ssv)
 +{
 +      uint8_t lv[GSM_MAX_SSVERSION + 1];
 +
 +      if (ssv->len > GSM_MAX_SSVERSION)
 +              return -EINVAL;
 +
 +      lv[0] = ssv->len;
 +      memcpy(lv + 1, ssv->info, ssv->len);
 +      msgb_tlv_put(msg, GSM48_IE_SS_VERS, lv[0], lv+1);
 +
 +      return 0;
 +}
 +
 +/* decode 'more data' does not require a function, because it has no value */
 +
 +/* encode 'more data' */
 +int gsm48_encode_more(struct msgb *msg)
 +{
 +      uint8_t *ie;
 +
 +      ie = msgb_put(msg, 1);
 +      ie[0] = GSM48_IE_MORE_DATA;
 +
 +      return 0;
 +}
 +
 +/* decode "Cell Channel Description" (10.5.2.1b) and other frequency lists */
 +int gsm48_decode_freq_list(struct gsm_sysinfo_freq *f, uint8_t *cd,
 +                         uint8_t len, uint8_t mask, uint8_t frqt)
 +{
 +      int i;
 +
 +      /* NOTES:
 +       *
 +       * The Range format uses "SMOD" computation.
 +       * e.g. "n SMOD m" equals "((n - 1) % m) + 1"
 +       * A cascade of multiple SMOD computations is simpified:
 +       * "(n SMOD m) SMOD o" equals "(((n - 1) % m) % o) + 1"
 +       *
 +       * The Range format uses 16 octets of data in SYSTEM INFORMATION.
 +       * When used in dedicated messages, the length can be less.
 +       * In this case the ranges are decoded for all frequencies that
 +       * fit in the block of given length.
 +       */
 +
 +      /* tabula rasa */
 +      for (i = 0; i < 1024; i++)
 +              f[i].mask &= ~frqt;
 +
 +      /* 00..XXX. */
 +      if ((cd[0] & 0xc0 & mask) == 0x00) {
 +              /* Bit map 0 format */
 +              if (len < 16)
 +                      return -EINVAL;
 +              for (i = 1; i <= 124; i++)
 +                      if ((cd[15 - ((i-1) >> 3)] & (1 << ((i-1) & 7))))
 +                              f[i].mask |= frqt;
 +
 +              return 0;
 +      }
 +
 +      /* 10..0XX. */
 +      if ((cd[0] & 0xc8 & mask) == 0x80) {
 +              /* Range 1024 format */
 +              uint16_t w[17]; /* 1..16 */
 +              struct gsm48_range_1024 *r = (struct gsm48_range_1024 *)cd;
 +
 +              if (len < 2)
 +                      return -EINVAL;
 +              memset(w, 0, sizeof(w));
 +              if (r->f0)
 +                      f[0].mask |= frqt;
 +              w[1] = (r->w1_hi << 8) | r->w1_lo;
 +              if (len >= 4)
 +                      w[2] = (r->w2_hi << 1) | r->w2_lo;
 +              if (len >= 5)
 +                      w[3] = (r->w3_hi << 2) | r->w3_lo;
 +              if (len >= 6)
 +                      w[4] = (r->w4_hi << 2) | r->w4_lo;
 +              if (len >= 7)
 +                      w[5] = (r->w5_hi << 2) | r->w5_lo;
 +              if (len >= 8)
 +                      w[6] = (r->w6_hi << 2) | r->w6_lo;
 +              if (len >= 9)
 +                      w[7] = (r->w7_hi << 2) | r->w7_lo;
 +              if (len >= 10)
 +                      w[8] = (r->w8_hi << 1) | r->w8_lo;
 +              if (len >= 10)
 +                      w[9] = r->w9;
 +              if (len >= 11)
 +                      w[10] = r->w10;
 +              if (len >= 12)
 +                      w[11] = (r->w11_hi << 6) | r->w11_lo;
 +              if (len >= 13)
 +                      w[12] = (r->w12_hi << 5) | r->w12_lo;
 +              if (len >= 14)
 +                      w[13] = (r->w13_hi << 4) | r->w13_lo;
 +              if (len >= 15)
 +                      w[14] = (r->w14_hi << 3) | r->w14_lo;
 +              if (len >= 16)
 +                      w[15] = (r->w15_hi << 2) | r->w15_lo;
 +              if (len >= 16)
 +                      w[16] = r->w16;
 +              if (w[1])
 +                      f[w[1]].mask |= frqt;
 +              if (w[2])
 +                      f[((w[1] - 512 + w[2] - 1) % 1023) + 1].mask |= frqt;
 +              if (w[3])
 +                      f[((w[1]       + w[3] - 1) % 1023) + 1].mask |= frqt;
 +              if (w[4])
 +                      f[((w[1] - 512 + ((w[2] - 256 + w[4] - 1) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[5])
++                      f[((w[1]       + ((w[3] - 256 + w[5] - 1) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[6])
 +                      f[((w[1] - 512 + ((w[2]       + w[6] - 1) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[7])
 +                      f[((w[1]       + ((w[3]       + w[7] - 1) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[8])
 +                      f[((w[1] - 512 + ((w[2] - 256 + ((w[4] - 128 + w[8] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[9])
 +                      f[((w[1]       + ((w[3] - 256 + ((w[5] - 128 + w[9] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[10])
 +                      f[((w[1] - 512 + ((w[2]       + ((w[6] - 128 + w[10] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[11])
 +                      f[((w[1]       + ((w[3]       + ((w[7] - 128 + w[11] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[12])
 +                      f[((w[1] - 512 + ((w[2] - 256 + ((w[4]       + w[12] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[13])
 +                      f[((w[1]       + ((w[3] - 256 + ((w[5]       + w[13] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[14])
 +                      f[((w[1] - 512 + ((w[2]       + ((w[6]       + w[14] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[15])
 +                      f[((w[1]       + ((w[3]       + ((w[7]       + w[15] - 1) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +              if (w[16])
 +                      f[((w[1] - 512 + ((w[2] - 256 + ((w[4] - 128 + ((w[8] - 64 + w[16] - 1) % 127)) % 255)) % 511)) % 1023) + 1].mask |= frqt;
 +
 +              return 0;
 +      }
 +      /* 10..100. */
 +      if ((cd[0] & 0xce & mask) == 0x88) {
 +              /* Range 512 format */
 +              uint16_t w[18]; /* 1..17 */
 +              struct gsm48_range_512 *r = (struct gsm48_range_512 *)cd;
 +
 +              if (len < 4)
 +                      return -EINVAL;
 +              memset(w, 0, sizeof(w));
 +              w[0] = (r->orig_arfcn_hi << 9) | (r->orig_arfcn_mid << 1) | r->orig_arfcn_lo;
 +              w[1] = (r->w1_hi << 2) | r->w1_lo;
 +              if (len >= 5)
 +                      w[2] = (r->w2_hi << 2) | r->w2_lo;
 +              if (len >= 6)
 +                      w[3] = (r->w3_hi << 2) | r->w3_lo;
 +              if (len >= 7)
 +                      w[4] = (r->w4_hi << 1) | r->w4_lo;
 +              if (len >= 7)
 +                      w[5] = r->w5;
 +              if (len >= 8)
 +                      w[6] = r->w6;
 +              if (len >= 9)
 +                      w[7] = (r->w7_hi << 6) | r->w7_lo;
 +              if (len >= 10)
 +                      w[8] = (r->w8_hi << 4) | r->w8_lo;
 +              if (len >= 11)
 +                      w[9] = (r->w9_hi << 2) | r->w9_lo;
 +              if (len >= 11)
 +                      w[10] = r->w10;
 +              if (len >= 12)
 +                      w[11] = r->w11;
 +              if (len >= 13)
 +                      w[12] = (r->w12_hi << 4) | r->w12_lo;
 +              if (len >= 14)
 +                      w[13] = (r->w13_hi << 2) | r->w13_lo;
 +              if (len >= 14)
 +                      w[14] = r->w14;
 +              if (len >= 15)
 +                      w[15] = r->w15;
 +              if (len >= 16)
 +                      w[16] = (r->w16_hi << 3) | r->w16_lo;
 +              if (len >= 16)
 +                      w[17] = r->w17;
 +              f[w[0]].mask |= frqt;
 +              if (w[1])
 +                      f[(w[0] + w[1]) % 1024].mask |= frqt;
 +              if (w[2])
 +                      f[(w[0] + ((w[1] - 256 + w[2] - 1) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[3])
 +                      f[(w[0] + ((w[1]       + w[3] - 1) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[4])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2] - 128 + w[4] - 1) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[5])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 128 + w[5] - 1) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[6])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2]       + w[6] - 1) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[7])
 +                      f[(w[0] + ((w[1]       + ((w[3]       + w[7] - 1) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[8])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2] - 128 + ((w[4] - 64 + w[8] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[9])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 128 + ((w[5] - 64 + w[9] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[10])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2]       + ((w[6] - 64 + w[10] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[11])
 +                      f[(w[0] + ((w[1]       + ((w[3]       + ((w[7] - 64 + w[11] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[12])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2] - 128 + ((w[4]      + w[12] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[13])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 128 + ((w[5]      + w[13] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[14])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2]       + ((w[6]      + w[14] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[15])
 +                      f[(w[0] + ((w[1]       + ((w[3]       + ((w[7]      + w[15] - 1) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[16])
 +                      f[(w[0] + ((w[1] - 256 + ((w[2] - 128 + ((w[4] - 64 + ((w[8] - 32 + w[16] - 1) % 63)) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +              if (w[17])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 128 + ((w[5] - 64 + ((w[9] - 32 + w[17] - 1) % 63)) % 127)) % 255)) % 511) + 1) % 1024].mask |= frqt;
 +
 +              return 0;
 +      }
 +      /* 10..101. */
 +      if ((cd[0] & 0xce & mask) == 0x8a) {
 +              /* Range 256 format */
 +              uint16_t w[22]; /* 1..21 */
 +              struct gsm48_range_256 *r = (struct gsm48_range_256 *)cd;
 +
 +              if (len < 4)
 +                      return -EINVAL;
 +              memset(w, 0, sizeof(w));
 +              w[0] = (r->orig_arfcn_hi << 9) | (r->orig_arfcn_mid << 1) | r->orig_arfcn_lo;
 +              w[1] = (r->w1_hi << 1) | r->w1_lo;
 +              if (len >= 4)
 +                      w[2] = r->w2;
 +              if (len >= 5)
 +                      w[3] = r->w3;
 +              if (len >= 6)
 +                      w[4] = (r->w4_hi << 5) | r->w4_lo;
 +              if (len >= 7)
 +                      w[5] = (r->w5_hi << 3) | r->w5_lo;
 +              if (len >= 8)
 +                      w[6] = (r->w6_hi << 1) | r->w6_lo;
 +              if (len >= 8)
 +                      w[7] = r->w7;
 +              if (len >= 9)
 +                      w[8] = (r->w8_hi << 4) | r->w8_lo;
 +              if (len >= 10)
 +                      w[9] = (r->w9_hi << 1) | r->w9_lo;
 +              if (len >= 10)
 +                      w[10] = r->w10;
 +              if (len >= 11)
 +                      w[11] = (r->w11_hi << 3) | r->w11_lo;
 +              if (len >= 11)
 +                      w[12] = r->w12;
 +              if (len >= 12)
 +                      w[13] = r->w13;
 +              if (len >= 13)
 +                      w[14] = r->w15;
 +              if (len >= 13)
 +                      w[15] = (r->w14_hi << 2) | r->w14_lo;
 +              if (len >= 14)
 +                      w[16] = (r->w16_hi << 3) | r->w16_lo;
 +              if (len >= 14)
 +                      w[17] = r->w17;
 +              if (len >= 15)
 +                      w[18] = r->w19;
 +              if (len >= 15)
 +                      w[19] = (r->w18_hi << 3) | r->w18_lo;
 +              if (len >= 16)
 +                      w[20] = (r->w20_hi << 3) | r->w20_lo;
 +              if (len >= 16)
 +                      w[21] = r->w21;
 +              f[w[0]].mask |= frqt;
 +              if (w[1])
 +                      f[(w[0] + w[1]) % 1024].mask |= frqt;
 +              if (w[2])
 +                      f[(w[0] + ((w[1] - 128 + w[2] - 1) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[3])
 +                      f[(w[0] + ((w[1]       + w[3] - 1) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[4])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2] - 64 + w[4] - 1) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[5])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 64 + w[5] - 1) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[6])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2]      + w[6] - 1) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[7])
 +                      f[(w[0] + ((w[1]       + ((w[3]      + w[7] - 1) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[8])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2] - 64 + ((w[4] - 32 + w[8] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[9])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 64 + ((w[5] - 32 + w[9] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[10])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2]      + ((w[6] - 32 + w[10] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[11])
 +                      f[(w[0] + ((w[1]       + ((w[3]      + ((w[7] - 32 + w[11] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[12])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2] - 64 + ((w[4]      + w[12] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[13])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 64 + ((w[5]      + w[13] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[14])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2]      + ((w[6]      + w[14] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[15])
 +                      f[(w[0] + ((w[1]       + ((w[3]      + ((w[7]      + w[15] - 1) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[16])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2] - 64 + ((w[4] - 32 + ((w[8] - 16 + w[16] - 1) % 31)) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[17])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 64 + ((w[5] - 32 + ((w[9] - 16 + w[17] - 1) % 31)) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[18])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2]      + ((w[6] - 32 + ((w[10] - 16 + w[18] - 1) % 31)) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[19])
 +                      f[(w[0] + ((w[1]       + ((w[3]      + ((w[7] - 32 + ((w[11] - 16 + w[19] - 1) % 31)) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[20])
 +                      f[(w[0] + ((w[1] - 128 + ((w[2] - 64 + ((w[4]      + ((w[12] - 16 + w[20] - 1) % 31)) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +              if (w[21])
 +                      f[(w[0] + ((w[1]       + ((w[3] - 64 + ((w[5]      + ((w[13] - 16 + w[21] - 1) % 31)) % 63)) % 127)) % 255) + 1) % 1024].mask |= frqt;
 +
 +              return 0;
 +      }
 +      /* 10..110. */
 +      if ((cd[0] & 0xce & mask) == 0x8c) {
 +              /* Range 128 format */
 +              uint16_t w[29]; /* 1..28 */
 +              struct gsm48_range_128 *r = (struct gsm48_range_128 *)cd;
 +
 +              if (len < 3)
 +                      return -EINVAL;
 +              memset(w, 0, sizeof(w));
 +              w[0] = (r->orig_arfcn_hi << 9) | (r->orig_arfcn_mid << 1) | r->orig_arfcn_lo;
 +              w[1] = r->w1;
 +              if (len >= 4)
 +                      w[2] = r->w2;
 +              if (len >= 5)
 +                      w[3] = (r->w3_hi << 4) | r->w3_lo;
 +              if (len >= 6)
 +                      w[4] = (r->w4_hi << 1) | r->w4_lo;
 +              if (len >= 6)
 +                      w[5] = r->w5;
 +              if (len >= 7)
 +                      w[6] = (r->w6_hi << 3) | r->w6_lo;
 +              if (len >= 7)
 +                      w[7] = r->w7;
 +              if (len >= 8)
 +                      w[8] = r->w8;
 +              if (len >= 8)
 +                      w[9] = r->w9;
 +              if (len >= 9)
 +                      w[10] = r->w10;
 +              if (len >= 9)
 +                      w[11] = r->w11;
 +              if (len >= 10)
 +                      w[12] = r->w12;
 +              if (len >= 10)
 +                      w[13] = r->w13;
 +              if (len >= 11)
 +                      w[14] = r->w14;
 +              if (len >= 11)
 +                      w[15] = r->w15;
 +              if (len >= 12)
 +                      w[16] = r->w16;
 +              if (len >= 12)
 +                      w[17] = r->w17;
 +              if (len >= 13)
 +                      w[18] = (r->w18_hi << 1) | r->w18_lo;
 +              if (len >= 13)
 +                      w[19] = r->w19;
 +              if (len >= 13)
 +                      w[20] = r->w20;
 +              if (len >= 14)
 +                      w[21] = (r->w21_hi << 2) | r->w21_lo;
 +              if (len >= 14)
 +                      w[22] = r->w22;
 +              if (len >= 14)
 +                      w[23] = r->w23;
 +              if (len >= 15)
 +                      w[24] = r->w24;
 +              if (len >= 15)
 +                      w[25] = r->w25;
 +              if (len >= 16)
 +                      w[26] = (r->w26_hi << 1) | r->w26_lo;
 +              if (len >= 16)
 +                      w[27] = r->w27;
 +              if (len >= 16)
 +                      w[28] = r->w28;
 +              f[w[0]].mask |= frqt;
 +              if (w[1])
 +                      f[(w[0] + w[1]) % 1024].mask |= frqt;
 +              if (w[2])
 +                      f[(w[0] + ((w[1] - 64 + w[2] - 1) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[3])
 +                      f[(w[0] + ((w[1]      + w[3] - 1) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[4])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + w[4] - 1) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[5])
 +                      f[(w[0] + ((w[1]      + ((w[3] - 32 + w[5] - 1) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[6])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2]      + w[6] - 1) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[7])
 +                      f[(w[0] + ((w[1]      + ((w[3]      + w[7] - 1) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[8])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + ((w[4] - 16 + w[8] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[9])
 +                      f[(w[0] + ((w[1]      + ((w[3] - 32 + ((w[5] - 16 + w[9] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[10])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2]      + ((w[6] - 16 + w[10] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[11])
 +                      f[(w[0] + ((w[1]      + ((w[3]      + ((w[7] - 16 + w[11] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[12])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + ((w[4]      + w[12] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[13])
 +                      f[(w[0] + ((w[1]      + ((w[3] - 32 + ((w[5]      + w[13] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[14])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2]      + ((w[6]      + w[14] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[15])
 +                      f[(w[0] + ((w[1]      + ((w[3]      + ((w[7]      + w[15] - 1) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[16])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + ((w[4] - 16 + ((w[8] - 8 + w[16] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[17])
 +                      f[(w[0] + ((w[1]      + ((w[3] - 32 + ((w[5] - 16 + ((w[9] - 8 + w[17] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[18])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2]      + ((w[6] - 16 + ((w[10] - 8 + w[18] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[19])
 +                      f[(w[0] + ((w[1]      + ((w[3]      + ((w[7] - 16 + ((w[11] - 8 + w[19] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[20])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + ((w[4]      + ((w[12] - 8 + w[20] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[21])
 +                      f[(w[0] + ((w[1]      + ((w[3] - 32 + ((w[5]      + ((w[13] - 8 + w[21] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[22])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2]      + ((w[6]      + ((w[14] - 8 + w[22] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[23])
 +                      f[(w[0] + ((w[1]      + ((w[3]      + ((w[7]      + ((w[15] - 8 + w[23] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[24])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + ((w[4] - 16 + ((w[8]     + w[24] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[25])
 +                      f[(w[0] + ((w[1]      + ((w[3] - 32 + ((w[5] - 16 + ((w[9]     + w[25] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[26])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2]      + ((w[6] - 16 + ((w[10]     + w[26] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[27])
 +                      f[(w[0] + ((w[1]      + ((w[3]      + ((w[7] - 16 + ((w[11]     + w[27] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +              if (w[28])
 +                      f[(w[0] + ((w[1] - 64 + ((w[2] - 32 + ((w[4]      + ((w[12]     + w[28] - 1) % 15)) % 31)) % 63)) % 127) + 1) % 1024].mask |= frqt;
 +
 +              return 0;
 +      }
 +      /* 10..111. */
 +      if ((cd[0] & 0xce & mask) == 0x8e) {
 +              /* Variable bitmap format (can be any length >= 3) */
 +              uint16_t orig = 0;
 +              struct gsm48_var_bit *r = (struct gsm48_var_bit *)cd;
 +
 +              if (len < 3)
 +                      return -EINVAL;
 +              orig = (r->orig_arfcn_hi << 9) | (r->orig_arfcn_mid << 1) | r->orig_arfcn_lo;
 +              f[orig].mask |= frqt;
 +              for (i = 1; 2 + (i >> 3) < len; i++)
 +                      if ((cd[2 + (i >> 3)] & (0x80 >> (i & 7))))
 +                              f[(orig + i) % 1024].mask |= frqt;
 +
 +              return 0;
 +      }
 +
 +      return 0;
 +}
index 44fea98,0000000..11d63ac
mode 100644,000000..100644
--- /dev/null
@@@ -1,671 -1,0 +1,671 @@@
-               .name = "DINP",
 +/* Debugging/Logging support code */
 +
 +/* (C) 2008-2010 by Harald Welte <laforge@gnumonks.org>
 + * (C) 2008 by Holger Hans Peter Freyther <zecke@selfish.org>
 + * All Rights Reserved
 + *
 + * 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, write to the Free Software Foundation, Inc.,
 + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 + *
 + */
 +
 +#include "../config.h"
 +
 +#include <stdarg.h>
 +#include <stdlib.h>
 +#include <stdio.h>
 +#include <string.h>
 +#include <ctype.h>
 +
 +#ifdef HAVE_STRINGS_H
 +#include <strings.h>
 +#endif
 +#include <time.h>
 +#include <errno.h>
 +
 +#include <osmocom/core/talloc.h>
 +#include <osmocom/core/utils.h>
 +#include <osmocom/core/logging.h>
 +
 +#include <osmocom/vty/logging.h>      /* for LOGGING_STR. */
 +
 +struct log_info *osmo_log_info;
 +
 +static struct log_context log_context;
 +static void *tall_log_ctx = NULL;
 +LLIST_HEAD(osmo_log_target_list);
 +
 +#define LOGLEVEL_DEFS 6       /* Number of loglevels.*/
 +
 +static const struct value_string loglevel_strs[LOGLEVEL_DEFS+1] = {
 +      { 0,            "EVERYTHING" },
 +      { LOGL_DEBUG,   "DEBUG" },
 +      { LOGL_INFO,    "INFO" },
 +      { LOGL_NOTICE,  "NOTICE" },
 +      { LOGL_ERROR,   "ERROR" },
 +      { LOGL_FATAL,   "FATAL" },
 +      { 0, NULL },
 +};
 +
 +#define INT2IDX(x)    (-1*(x)-1)
 +static const struct log_info_cat internal_cat[OSMO_NUM_DLIB] = {
 +      [INT2IDX(DLGLOBAL)] = { /* -1 becomes 0 */
 +              .name = "DLGLOBAL",
 +              .description = "Library-internal global log family",
 +              .loglevel = LOGL_NOTICE,
 +              .enabled = 1,
 +      },
 +      [INT2IDX(DLLAPDM)] = {  /* -2 becomes 1 */
 +              .name = "DLLAPDM",
 +              .description = "LAPDm in libosmogsm",
 +              .loglevel = LOGL_NOTICE,
 +              .enabled = 1,
 +      },
 +      [INT2IDX(DLINP)] = {
-               .name = "DMUX",
++              .name = "DLINP",
 +              .description = "A-bis Intput Subsystem",
 +              .loglevel = LOGL_NOTICE,
 +              .enabled = 1,
 +      },
 +      [INT2IDX(DLMUX)] = {
-               .name = "DMI",
++              .name = "DLMUX",
 +              .description = "A-bis B-Subchannel TRAU Frame Multiplex",
 +              .loglevel = LOGL_NOTICE,
 +              .enabled = 1,
 +      },
 +      [INT2IDX(DLMI)] = {
-               .name = "DMIB",
++              .name = "DLMI",
 +              .description = "A-bis Input Driver for Signalling",
 +              .enabled = 0, .loglevel = LOGL_NOTICE,
 +      },
 +      [INT2IDX(DLMIB)] = {
++              .name = "DLMIB",
 +              .description = "A-bis Input Driver for B-Channels (voice)",
 +              .enabled = 0, .loglevel = LOGL_NOTICE,
 +      },
 +};
 +
 +/* You have to keep this in sync with the structure loglevel_strs. */
 +const char *loglevel_descriptions[LOGLEVEL_DEFS+1] = {
 +      "Log simply everything",
 +      "Log debug messages and higher levels",
 +      "Log informational messages and higher levels",
 +      "Log noticable messages and higher levels",
 +      "Log error messages and higher levels",
 +      "Log only fatal messages",
 +      NULL,
 +};
 +
 +/* special magic for negative (library-internal) log subsystem numbers */
 +static int subsys_lib2index(int subsys)
 +{
 +      return (subsys * -1) + (osmo_log_info->num_cat_user-1);
 +}
 +
 +int log_parse_level(const char *lvl)
 +{
 +      return get_string_value(loglevel_strs, lvl);
 +}
 +
 +const char *log_level_str(unsigned int lvl)
 +{
 +      return get_value_string(loglevel_strs, lvl);
 +}
 +
 +int log_parse_category(const char *category)
 +{
 +      int i;
 +
 +      for (i = 0; i < osmo_log_info->num_cat; ++i) {
 +              if (osmo_log_info->cat[i].name == NULL)
 +                      continue;
 +              if (!strcasecmp(osmo_log_info->cat[i].name+1, category))
 +                      return i;
 +      }
 +
 +      return -EINVAL;
 +}
 +
 +/*
 + * Parse the category mask.
 + * The format can be this: category1:category2:category3
 + * or category1,2:category2,3:...
 + */
 +void log_parse_category_mask(struct log_target* target, const char *_mask)
 +{
 +      int i = 0;
 +      char *mask = strdup(_mask);
 +      char *category_token = NULL;
 +
 +      /* Disable everything to enable it afterwards */
 +      for (i = 0; i < osmo_log_info->num_cat; ++i)
 +              target->categories[i].enabled = 0;
 +
 +      category_token = strtok(mask, ":");
 +      do {
 +              for (i = 0; i < osmo_log_info->num_cat; ++i) {
 +                      char* colon = strstr(category_token, ",");
 +                      int length = strlen(category_token);
 +
 +                      if (!osmo_log_info->cat[i].name)
 +                              continue;
 +
 +                      if (colon)
 +                          length = colon - category_token;
 +
 +                      if (strncasecmp(osmo_log_info->cat[i].name,
 +                                      category_token, length) == 0) {
 +                              int level = 0;
 +
 +                              if (colon)
 +                                      level = atoi(colon+1);
 +
 +                              target->categories[i].enabled = 1;
 +                              target->categories[i].loglevel = level;
 +                      }
 +              }
 +      } while ((category_token = strtok(NULL, ":")));
 +
 +      free(mask);
 +}
 +
 +static const char* color(int subsys)
 +{
 +      if (subsys < osmo_log_info->num_cat)
 +              return osmo_log_info->cat[subsys].color;
 +
 +      return NULL;
 +}
 +
 +static void _output(struct log_target *target, unsigned int subsys,
 +                  unsigned int level, char *file, int line, int cont,
 +                  const char *format, va_list ap)
 +{
 +      char buf[4096];
 +      int ret, len = 0, offset = 0, rem = sizeof(buf);
 +
 +      /* are we using color */
 +      if (target->use_color) {
 +              const char *c = color(subsys);
 +              if (c) {
 +                      ret = snprintf(buf + offset, rem, "%s", color(subsys));
 +                      if (ret < 0)
 +                              goto err;
 +                      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +              }
 +      }
 +      if (!cont) {
 +              if (target->print_timestamp) {
 +                      char *timestr;
 +                      time_t tm;
 +                      tm = time(NULL);
 +                      timestr = ctime(&tm);
 +                      timestr[strlen(timestr)-1] = '\0';
 +                      ret = snprintf(buf + offset, rem, "%s ", timestr);
 +                      if (ret < 0)
 +                              goto err;
 +                      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +              }
 +              ret = snprintf(buf + offset, rem, "<%4.4x> %s:%d ",
 +                              subsys, file, line);
 +              if (ret < 0)
 +                      goto err;
 +              OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +      }
 +      ret = vsnprintf(buf + offset, rem, format, ap);
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +
 +      ret = snprintf(buf + offset, rem, "%s",
 +                      target->use_color ? "\033[0;m" : "");
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +err:
 +      buf[sizeof(buf)-1] = '\0';
 +      target->output(target, level, buf);
 +}
 +
 +static void _logp(int subsys, int level, char *file, int line,
 +                int cont, const char *format, va_list ap)
 +{
 +      struct log_target *tar;
 +
 +      if (subsys < 0)
 +              subsys = subsys_lib2index(subsys);
 +
 +      if (subsys > osmo_log_info->num_cat)
 +              subsys = DLGLOBAL;
 +
 +      llist_for_each_entry(tar, &osmo_log_target_list, entry) {
 +              struct log_category *category;
 +              int output = 0;
 +              va_list bp;
 +
 +              category = &tar->categories[subsys];
 +              /* subsystem is not supposed to be logged */
 +              if (!category->enabled)
 +                      continue;
 +
 +              /* Check the global log level */
 +              if (tar->loglevel != 0 && level < tar->loglevel)
 +                      continue;
 +
 +              /* Check the category log level */
 +              if (tar->loglevel == 0 && category->loglevel != 0 &&
 +                  level < category->loglevel)
 +                      continue;
 +
 +              /* Apply filters here... if that becomes messy we will
 +               * need to put filters in a list and each filter will
 +               * say stop, continue, output */
 +              if ((tar->filter_map & LOG_FILTER_ALL) != 0)
 +                      output = 1;
 +              else if (osmo_log_info->filter_fn)
 +                      output = osmo_log_info->filter_fn(&log_context,
 +                                                     tar);
 +              if (!output)
 +                      continue;
 +
 +              /* According to the manpage, vsnprintf leaves the value of ap
 +               * in undefined state. Since _output uses vsnprintf and it may
 +               * be called several times, we have to pass a copy of ap. */
 +              va_copy(bp, ap);
 +              _output(tar, subsys, level, file, line, cont, format, bp);
 +              va_end(bp);
 +      }
 +}
 +
 +void logp(int subsys, char *file, int line, int cont,
 +        const char *format, ...)
 +{
 +      va_list ap;
 +
 +      va_start(ap, format);
 +      _logp(subsys, LOGL_DEBUG, file, line, cont, format, ap);
 +      va_end(ap);
 +}
 +
 +void logp2(int subsys, unsigned int level, char *file, int line, int cont, const char *format, ...)
 +{
 +      va_list ap;
 +
 +      va_start(ap, format);
 +      _logp(subsys, level, file, line, cont, format, ap);
 +      va_end(ap);
 +}
 +
 +void log_add_target(struct log_target *target)
 +{
 +      llist_add_tail(&target->entry, &osmo_log_target_list);
 +}
 +
 +void log_del_target(struct log_target *target)
 +{
 +      llist_del(&target->entry);
 +}
 +
 +void log_reset_context(void)
 +{
 +      memset(&log_context, 0, sizeof(log_context));
 +}
 +
 +int log_set_context(uint8_t ctx_nr, void *value)
 +{
 +      if (ctx_nr > LOG_MAX_CTX)
 +              return -EINVAL;
 +
 +      log_context.ctx[ctx_nr] = value;
 +
 +      return 0;
 +}
 +
 +void log_set_all_filter(struct log_target *target, int all)
 +{
 +      if (all)
 +              target->filter_map |= LOG_FILTER_ALL;
 +      else
 +              target->filter_map &= ~LOG_FILTER_ALL;
 +}
 +
 +void log_set_use_color(struct log_target *target, int use_color)
 +{
 +      target->use_color = use_color;
 +}
 +
 +void log_set_print_timestamp(struct log_target *target, int print_timestamp)
 +{
 +      target->print_timestamp = print_timestamp;
 +}
 +
 +void log_set_log_level(struct log_target *target, int log_level)
 +{
 +      target->loglevel = log_level;
 +}
 +
 +void log_set_category_filter(struct log_target *target, int category,
 +                             int enable, int level)
 +{
 +      if (category >= osmo_log_info->num_cat)
 +              return;
 +      target->categories[category].enabled = !!enable;
 +      target->categories[category].loglevel = level;
 +}
 +
 +static void _file_output(struct log_target *target, unsigned int level,
 +                       const char *log)
 +{
 +      fprintf(target->tgt_file.out, "%s", log);
 +      fflush(target->tgt_file.out);
 +}
 +
 +struct log_target *log_target_create(void)
 +{
 +      struct log_target *target;
 +      unsigned int i;
 +
 +      target = talloc_zero(tall_log_ctx, struct log_target);
 +      if (!target)
 +              return NULL;
 +
 +      target->categories = talloc_zero_array(target, 
 +                                              struct log_category,
 +                                              osmo_log_info->num_cat);
 +      if (!target->categories) {
 +              talloc_free(target);
 +              return NULL;
 +      }
 +
 +      INIT_LLIST_HEAD(&target->entry);
 +
 +      /* initialize the per-category enabled/loglevel from defaults */
 +      for (i = 0; i < osmo_log_info->num_cat; i++) {
 +              struct log_category *cat = &target->categories[i];
 +              cat->enabled = osmo_log_info->cat[i].enabled;
 +              cat->loglevel = osmo_log_info->cat[i].loglevel;
 +      }
 +
 +      /* global settings */
 +      target->use_color = 1;
 +      target->print_timestamp = 0;
 +
 +      /* global log level */
 +      target->loglevel = 0;
 +      return target;
 +}
 +
 +struct log_target *log_target_create_stderr(void)
 +{
 +/* since C89/C99 says stderr is a macro, we can safely do this! */
 +#ifdef stderr
 +      struct log_target *target;
 +
 +      target = log_target_create();
 +      if (!target)
 +              return NULL;
 +
 +      target->type = LOG_TGT_TYPE_STDERR;
 +      target->tgt_file.out = stderr;
 +      target->output = _file_output;
 +      return target;
 +#else
 +      return NULL;
 +#endif /* stderr */
 +}
 +
 +struct log_target *log_target_create_file(const char *fname)
 +{
 +      struct log_target *target;
 +
 +      target = log_target_create();
 +      if (!target)
 +              return NULL;
 +
 +      target->type = LOG_TGT_TYPE_FILE;
 +      target->tgt_file.out = fopen(fname, "a");
 +      if (!target->tgt_file.out)
 +              return NULL;
 +
 +      target->output = _file_output;
 +
 +      target->tgt_file.fname = talloc_strdup(target, fname);
 +
 +      return target;
 +}
 +
 +struct log_target *log_target_find(int type, const char *fname)
 +{
 +      struct log_target *tgt;
 +
 +      llist_for_each_entry(tgt, &osmo_log_target_list, entry) {
 +              if (tgt->type != type)
 +                      continue;
 +              if (tgt->type == LOG_TGT_TYPE_FILE) {
 +                      if (!strcmp(fname, tgt->tgt_file.fname))
 +                              return tgt;
 +              } else
 +                      return tgt;
 +      }
 +      return NULL;
 +}
 +
 +void log_target_destroy(struct log_target *target)
 +{
 +
 +      /* just in case, to make sure we don't have any references */
 +      log_del_target(target);
 +
 +      if (target->output == &_file_output) {
 +/* since C89/C99 says stderr is a macro, we can safely do this! */
 +#ifdef stderr
 +              /* don't close stderr */
 +              if (target->tgt_file.out != stderr)
 +#endif
 +              {
 +                      fclose(target->tgt_file.out);
 +                      target->tgt_file.out = NULL;
 +              }
 +      }
 +
 +      talloc_free(target);
 +}
 +
 +/* close and re-open a log file (for log file rotation) */
 +int log_target_file_reopen(struct log_target *target)
 +{
 +      fclose(target->tgt_file.out);
 +
 +      target->tgt_file.out = fopen(target->tgt_file.fname, "a");
 +      if (!target->tgt_file.out)
 +              return -errno;
 +
 +      /* we assume target->output already to be set */
 +
 +      return 0;
 +}
 +
 +/* This generates the logging command string for VTY. */
 +const char *log_vty_command_string(const struct log_info *unused_info)
 +{
 +      struct log_info *info = osmo_log_info;
 +      int len = 0, offset = 0, ret, i, rem;
 +      int size = strlen("logging level () ()") + 1;
 +      char *str;
 +
 +      for (i = 0; i < info->num_cat; i++) {
 +              if (info->cat[i].name == NULL)
 +                      continue;
 +              size += strlen(info->cat[i].name) + 1;
 +      }
 +
 +      for (i = 0; i < LOGLEVEL_DEFS; i++)
 +              size += strlen(loglevel_strs[i].str) + 1;
 +
 +      rem = size;
 +      str = talloc_zero_size(tall_log_ctx, size);
 +      if (!str)
 +              return NULL;
 +
 +      ret = snprintf(str + offset, rem, "logging level (all|");
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +
 +      for (i = 0; i < info->num_cat; i++) {
 +              if (info->cat[i].name) {
 +                      int j, name_len = strlen(info->cat[i].name)+1;
 +                      char name[name_len];
 +
 +                      for (j = 0; j < name_len; j++)
 +                              name[j] = tolower(info->cat[i].name[j]);
 +
 +                      name[name_len-1] = '\0';
 +                      ret = snprintf(str + offset, rem, "%s|", name+1);
 +                      if (ret < 0)
 +                              goto err;
 +                      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +              }
 +      }
 +      offset--;       /* to remove the trailing | */
 +      rem++;
 +
 +      ret = snprintf(str + offset, rem, ") (");
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +
 +      for (i = 0; i < LOGLEVEL_DEFS; i++) {
 +              int j, loglevel_str_len = strlen(loglevel_strs[i].str)+1;
 +              char loglevel_str[loglevel_str_len];
 +
 +              for (j = 0; j < loglevel_str_len; j++)
 +                      loglevel_str[j] = tolower(loglevel_strs[i].str[j]);
 +
 +              loglevel_str[loglevel_str_len-1] = '\0';
 +              ret = snprintf(str + offset, rem, "%s|", loglevel_str);
 +              if (ret < 0)
 +                      goto err;
 +              OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +      }
 +      offset--;       /* to remove the trailing | */
 +      rem++;
 +
 +      ret = snprintf(str + offset, rem, ")");
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +err:
 +      str[size-1] = '\0';
 +      return str;
 +}
 +
 +/* This generates the logging command description for VTY. */
 +const char *log_vty_command_description(const struct log_info *unused_info)
 +{
 +      struct log_info *info = osmo_log_info;
 +      char *str;
 +      int i, ret, len = 0, offset = 0, rem;
 +      unsigned int size =
 +              strlen(LOGGING_STR
 +                     "Set the log level for a specified category\n") + 1;
 +
 +      for (i = 0; i < info->num_cat; i++) {
 +              if (info->cat[i].name == NULL)
 +                      continue;
 +              size += strlen(info->cat[i].description) + 1;
 +      }
 +
 +      for (i = 0; i < LOGLEVEL_DEFS; i++)
 +              size += strlen(loglevel_descriptions[i]) + 1;
 +
 +      size += strlen("Global setting for all subsystems") + 1;
 +      rem = size;
 +      str = talloc_zero_size(tall_log_ctx, size);
 +      if (!str)
 +              return NULL;
 +
 +      ret = snprintf(str + offset, rem, LOGGING_STR
 +                      "Set the log level for a specified category\n");
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +
 +      ret = snprintf(str + offset, rem,
 +                      "Global setting for all subsystems\n");
 +      if (ret < 0)
 +              goto err;
 +      OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +
 +      for (i = 0; i < info->num_cat; i++) {
 +              if (info->cat[i].name == NULL)
 +                      continue;
 +              ret = snprintf(str + offset, rem, "%s\n",
 +                              info->cat[i].description);
 +              if (ret < 0)
 +                      goto err;
 +              OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +      }
 +      for (i = 0; i < LOGLEVEL_DEFS; i++) {
 +              ret = snprintf(str + offset, rem, "%s\n",
 +                              loglevel_descriptions[i]);
 +              if (ret < 0)
 +                      goto err;
 +              OSMO_SNPRINTF_RET(ret, rem, offset, len);
 +      }
 +err:
 +      str[size-1] = '\0';
 +      return str;
 +}
 +
 +int log_init(const struct log_info *inf, void *ctx)
 +{
 +      int i;
 +
 +      tall_log_ctx = talloc_named_const(ctx, 1, "logging");
 +      if (!tall_log_ctx)
 +              return -ENOMEM;
 +
 +      osmo_log_info = talloc_zero(tall_log_ctx, struct log_info);
 +      if (!osmo_log_info)
 +              return -ENOMEM;
 +
 +      osmo_log_info->num_cat_user = inf->num_cat;
 +      /* total number = number of user cat + library cat */
 +      osmo_log_info->num_cat = inf->num_cat + ARRAY_SIZE(internal_cat);
 +
 +      osmo_log_info->cat = talloc_zero_array(osmo_log_info,
 +                                      struct log_info_cat,
 +                                      osmo_log_info->num_cat);
 +      if (!osmo_log_info->cat) {
 +              talloc_free(osmo_log_info);
 +              osmo_log_info = NULL;
 +              return -ENOMEM;
 +      }
 +
 +      /* copy over the user part */
 +      for (i = 0; i < inf->num_cat; i++) {
 +              memcpy(&osmo_log_info->cat[i], &inf->cat[i],
 +                      sizeof(struct log_info_cat));
 +      }
 +
 +      /* copy over the library part */
 +      for (i = 0; i < ARRAY_SIZE(internal_cat); i++) {
 +              unsigned int cn = osmo_log_info->num_cat_user + i;
 +              memcpy(&osmo_log_info->cat[cn],
 +                      &internal_cat[i], sizeof(struct log_info_cat));
 +      }
 +
 +      return 0;
 +}