C определить макро-функцию в заголовочном файле, указывающую на внешнюю функцию - PullRequest
0 голосов
/ 26 августа 2018

У меня есть вопрос об определении макроса функции в C (я новичок): у меня есть библиотека COTS lib.a, которая использует функцию strtok(), но мое приложение CERT поддерживает только strtok_r, поэтому яполучить ошибку во время компиляции.

Как я могу определить в заголовочном файле, что функция strtok должна быть переопределена с strtok_r?

Я пытался что-то вроде этого, но я получаюошибки:

extern char *strtok_r(char *s1, const char *s2, char **saveptr);
#define strtok(s1,s2) strtok_r(s1,s2,saveptr)

Какой самый лучший и чистый способ достижения результата?

Большое спасибо за ваш вклад.

Ответы [ 2 ]

0 голосов
/ 26 августа 2018

Если я вас правильно понял:

  1. У вас нет функции strtok.
  2. Вы используете библиотеку libother.a (просто, чтобы дать ей имя), которая внутренне использует функцию strtok -.
  3. У вас нет исходного кода этой библиотеки.

Роб на правильном пути, но, я думаю, все пошло не так ...

Вы можете создать свою собственную функцию strtok, обернув функцию strtok_r следующим образом:

Вставьте эту функцию в один из ваших исходных файлов:

char *strtok(char *s1, const char *delim) {
    static char* saveptr = 0;
    return strtok_r(s1, delim, &saveptr);
}

Затем компилируем и связываем ваш код обычным образом, как

gcc your_code_file.c -lother -o your_binary

должен сделать. Однако есть причина, по которой вы должны предпочесть использовать strtok_r вместо strtok: strtok может анализировать только одну строку за раз, strtok_r позволяет одновременно анализировать произвольное количество различных строк.

С «взломом» выше вы теряете это преимущество strtok_r.

0 голосов
/ 26 августа 2018

Вы не можете переопределить strtok_r как strtok, потому что при разном количестве аргументов они работают по-разному.

char * strtok (char * restrict, const char * restrict);

char * strtok_r (char *, const char *, char **);

Если у вас нет strtok_r, тогда вы можете написать его.

Комментатор объяснил, что вы хотите реализовать strtok из strtok_r, поэтому я включаю и код для этого.

Это strtok_r () из: WoBoq

/* Reentrant string tokenizer.  Generic version.
   Copyright (C) 1991-2018 Free Software Foundation, Inc.
   This file is part of the GNU C Library.

   The GNU C Library is free software; you can redistribute it and/or
   modify it under the terms of the GNU Lesser General Public
   License as published by the Free Software Foundation; either
   version 2.1 of the License, or (at your option) any later version.

   The GNU C Library 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
   Lesser General Public License for more details.

   You should have received a copy of the GNU Lesser General Public
   License along with the GNU C Library; if not, see
   <http://www.gnu.org/licenses/>.  */

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif

#include <string.h>

#ifndef _LIBC
/* Get specification.  */
# include "strtok_r.h"
# define __strtok_r strtok_r
#endif

/* Parse S into tokens separated by characters in DELIM.
   If S is NULL, the saved pointer in SAVE_PTR is used as
   the next starting point.  For example:
        char s[] = "-abc-=-def";
        char *sp;
        x = strtok_r(s, "-", &sp);        // x = "abc", sp = "=-def"
        x = strtok_r(NULL, "-=", &sp);        // x = "def", sp = NULL
        x = strtok_r(NULL, "=", &sp);        // x = NULL
                // s = "abc\0-def\0"
*/

char *
__strtok_r (char *s, const char *delim, char **save_ptr)
{
  char *end;

  if (s == NULL)
    s = *save_ptr;

  if (*s == '\0')
    {
      *save_ptr = s;
      return NULL;
    }

  /* Scan leading delimiters.  */
  s += strspn (s, delim);
  if (*s == '\0')
    {
      *save_ptr = s;
      return NULL;
    }

  /* Find the end of the token.  */
  end = s + strcspn (s, delim);
  if (*end == '\0')
    {
      *save_ptr = end;
      return s;
    }

  /* Terminate the token and make *SAVE_PTR point past it.  */
  *end = '\0';
  *save_ptr = end + 1;
  return s;
}
#ifdef weak_alias
libc_hidden_def (__strtok_r)
weak_alias (__strtok_r, strtok_r)
#endif

Это strtok () из: glibc / blob / master / string / strtok.c

/* Copyright (C) 1991-2018 Free Software Foundation, Inc.
   This file is part of the GNU C Library.

   The GNU C Library is free software; you can redistribute it and/or
   modify it under the terms of the GNU Lesser General Public
   License as published by the Free Software Foundation; either
   version 2.1 of the License, or (at your option) any later version.

   The GNU C Library 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
   Lesser General Public License for more details.
   You should have received a copy of the GNU Lesser General Public
   License along with the GNU C Library; if not, see
   <http://www.gnu.org/licenses/>.  */

#include <string.h>

/* Parse S into tokens separated by characters in DELIM.
   If S is NULL, the last string strtok() was called with is
   used.  For example:
    char s[] = "-abc-=-def";
    x = strtok(s, "-");     // x = "abc"
    x = strtok(NULL, "-=");     // x = "def"
    x = strtok(NULL, "=");      // x = NULL
        // s = "abc\0=-def\0"
*/
char *
strtok (char *s, const char *delim)
{
  static char *olds;
  return __strtok_r (s, delim, &olds);
}

Это один пример реализации каждой функции. Есть много доступных источников компилятора, содержащих библиотеку. Если эта лицензия не соответствует вашим требованиям, попробуйте найти другую с условиями, отвечающими вашим потребностям, или напишите свою собственную функцию.

Просто запишите этот источник в отдельный файл и добавьте его в список ссылок.

Проблема с попыткой переопределить эти функции друг для друга объясняется и демонстрируется на этой веб-странице GeeksForGeeks: « функции strtok () и strtok_r () в C с примерами ». Функция strtok_r требует других параметров для последующих вызовов, чем те, которые предусмотрены для первого вызова. С глобальным не удобно делать определение, которое может решить, устанавливать или тестировать.

...