mirror of
				https://github.com/MariaDB/server.git
				synced 2025-11-04 12:56:14 +01:00 
			
		
		
		
	This commit is the final batch of #3360’s `ATTRIBUTE_FORMAT` process, covering various insignificant (as in, requires few-to-no changes in addition to gaining this attribute) functions. One of the main focus of this PR is to enable GCC `-Wformat` (by tagging `ATTRIBUTE_FORMAT`) on ALL `my_snprintf` utilities. To be throughout, functions that delegate to `my_vsnprintf` must also inherit this attribute because `-Wformat` doesn’t trace argument across call stacks.
		
			
				
	
	
		
			655 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			655 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
Copyright (c) 2007, Antony T Curtis
 | 
						|
All rights reserved.
 | 
						|
 | 
						|
Redistribution and use in source and binary forms, with or without
 | 
						|
modification, are permitted provided that the following conditions are
 | 
						|
met:
 | 
						|
 | 
						|
    * Redistributions of source code must retain the above copyright
 | 
						|
notice, this list of conditions and the following disclaimer.
 | 
						|
 | 
						|
    * Neither the name of FederatedX nor the names of its
 | 
						|
contributors may be used to endorse or promote products derived from
 | 
						|
this software without specific prior written permission.
 | 
						|
 | 
						|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 | 
						|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 | 
						|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 | 
						|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 | 
						|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | 
						|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 | 
						|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 | 
						|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 | 
						|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | 
						|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | 
						|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | 
						|
*/
 | 
						|
 | 
						|
 | 
						|
#define MYSQL_SERVER 1
 | 
						|
#include <my_global.h>
 | 
						|
#include "sql_priv.h"
 | 
						|
#include <mysqld_error.h>
 | 
						|
#include <mysql.h>
 | 
						|
 | 
						|
#include "ha_federatedx.h"
 | 
						|
 | 
						|
#include "m_string.h"
 | 
						|
#include "mysqld_error.h"
 | 
						|
#include "sql_servers.h"
 | 
						|
 | 
						|
#define SAVEPOINT_REALIZED  1
 | 
						|
#define SAVEPOINT_RESTRICT  2
 | 
						|
#define SAVEPOINT_EMITTED 4
 | 
						|
 | 
						|
 | 
						|
typedef struct federatedx_savepoint
 | 
						|
{
 | 
						|
  ulong level;
 | 
						|
  uint  flags;
 | 
						|
} SAVEPT;
 | 
						|
 | 
						|
struct mysql_position
 | 
						|
{
 | 
						|
  MYSQL_RES* result;
 | 
						|
  MYSQL_ROW_OFFSET offset;
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
class federatedx_io_mysql :public federatedx_io
 | 
						|
{
 | 
						|
  MYSQL mysql; /* MySQL connection */
 | 
						|
  DYNAMIC_ARRAY savepoints;
 | 
						|
  bool requested_autocommit;
 | 
						|
  bool actual_autocommit;
 | 
						|
 | 
						|
  int actual_query(const char *buffer, size_t length);
 | 
						|
  bool test_all_restrict() const;
 | 
						|
public:
 | 
						|
  federatedx_io_mysql(FEDERATEDX_SERVER *);
 | 
						|
  ~federatedx_io_mysql() override;
 | 
						|
 | 
						|
  // 1st arg is the implicit `this`
 | 
						|
  int simple_query(const char *fmt, ...) ATTRIBUTE_FORMAT(printf, 2, 3);
 | 
						|
  int query(const char *buffer, size_t length) override;
 | 
						|
  FEDERATEDX_IO_RESULT *store_result() override;
 | 
						|
 | 
						|
  size_t max_query_size() const override;
 | 
						|
 | 
						|
  my_ulonglong affected_rows() const override;
 | 
						|
  my_ulonglong last_insert_id() const override;
 | 
						|
 | 
						|
  int error_code() override;
 | 
						|
  const char *error_str() override;
 | 
						|
 | 
						|
  void reset() override;
 | 
						|
  int commit() override;
 | 
						|
  int rollback() override;
 | 
						|
 | 
						|
  int savepoint_set(ulong sp) override;
 | 
						|
  ulong savepoint_release(ulong sp) override;
 | 
						|
  ulong savepoint_rollback(ulong sp) override;
 | 
						|
  void savepoint_restrict(ulong sp) override;
 | 
						|
 | 
						|
  ulong last_savepoint() const override;
 | 
						|
  ulong actual_savepoint() const override;
 | 
						|
  bool is_autocommit() const override;
 | 
						|
 | 
						|
  bool table_metadata(ha_statistics *stats, const char *table_name,
 | 
						|
                      uint table_name_length, uint flag) override;
 | 
						|
 | 
						|
  /* resultset operations */
 | 
						|
 | 
						|
  void free_result(FEDERATEDX_IO_RESULT *io_result) override;
 | 
						|
  unsigned int get_num_fields(FEDERATEDX_IO_RESULT *io_result) override;
 | 
						|
  my_ulonglong get_num_rows(FEDERATEDX_IO_RESULT *io_result) override;
 | 
						|
  FEDERATEDX_IO_ROW *fetch_row(FEDERATEDX_IO_RESULT *io_result,
 | 
						|
                                       FEDERATEDX_IO_ROWS **current= NULL) override;
 | 
						|
  ulong *fetch_lengths(FEDERATEDX_IO_RESULT *io_result) override;
 | 
						|
  const char *get_column_data(FEDERATEDX_IO_ROW *row,
 | 
						|
                                      unsigned int column) override;
 | 
						|
  bool is_column_null(const FEDERATEDX_IO_ROW *row,
 | 
						|
                              unsigned int column) const override;
 | 
						|
 | 
						|
  size_t get_ref_length() const override;
 | 
						|
  void mark_position(FEDERATEDX_IO_RESULT *io_result,
 | 
						|
                             void *ref, FEDERATEDX_IO_ROWS *current) override;
 | 
						|
  int seek_position(FEDERATEDX_IO_RESULT **io_result,
 | 
						|
                            const void *ref) override;
 | 
						|
  void set_thd(void *thd) override;
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
federatedx_io *instantiate_io_mysql(MEM_ROOT *server_root,
 | 
						|
                                    FEDERATEDX_SERVER *server)
 | 
						|
{
 | 
						|
  return new (server_root) federatedx_io_mysql(server);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
federatedx_io_mysql::federatedx_io_mysql(FEDERATEDX_SERVER *aserver)
 | 
						|
  : federatedx_io(aserver),
 | 
						|
    requested_autocommit(TRUE), actual_autocommit(TRUE)
 | 
						|
{
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::federatedx_io_mysql");
 | 
						|
 | 
						|
  bzero(&mysql, sizeof(MYSQL));
 | 
						|
  bzero(&savepoints, sizeof(DYNAMIC_ARRAY));
 | 
						|
 | 
						|
  my_init_dynamic_array(PSI_INSTRUMENT_ME, &savepoints, sizeof(SAVEPT), 16, 16, MYF(0));
 | 
						|
  
 | 
						|
  DBUG_VOID_RETURN;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
federatedx_io_mysql::~federatedx_io_mysql()
 | 
						|
{
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::~federatedx_io_mysql");
 | 
						|
 | 
						|
  mysql_close(&mysql);
 | 
						|
  delete_dynamic(&savepoints);
 | 
						|
 | 
						|
  DBUG_VOID_RETURN;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void federatedx_io_mysql::reset()
 | 
						|
{
 | 
						|
  reset_dynamic(&savepoints);
 | 
						|
  set_active(FALSE);
 | 
						|
  
 | 
						|
  requested_autocommit= TRUE;
 | 
						|
  mysql.reconnect= 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int federatedx_io_mysql::commit()
 | 
						|
{
 | 
						|
  int error= 0;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::commit");
 | 
						|
  
 | 
						|
  if (!actual_autocommit && (error= actual_query("COMMIT", 6)))
 | 
						|
    rollback();
 | 
						|
  
 | 
						|
  reset();
 | 
						|
  
 | 
						|
  DBUG_RETURN(error);
 | 
						|
}
 | 
						|
 | 
						|
int federatedx_io_mysql::rollback()
 | 
						|
{
 | 
						|
  int error= 0;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::rollback");
 | 
						|
  
 | 
						|
  if (!actual_autocommit)
 | 
						|
    error= actual_query("ROLLBACK", 8);
 | 
						|
  else
 | 
						|
    error= ER_WARNING_NOT_COMPLETE_ROLLBACK;
 | 
						|
 | 
						|
  reset();
 | 
						|
  
 | 
						|
  DBUG_RETURN(error);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
ulong federatedx_io_mysql::last_savepoint() const
 | 
						|
{
 | 
						|
  SAVEPT *savept= NULL;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::last_savepoint");
 | 
						|
 | 
						|
  if (savepoints.elements)
 | 
						|
    savept= dynamic_element(&savepoints, savepoints.elements - 1, SAVEPT *);
 | 
						|
 | 
						|
  DBUG_RETURN(savept ? savept->level : 0);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
ulong federatedx_io_mysql::actual_savepoint() const
 | 
						|
{
 | 
						|
  SAVEPT *savept= NULL;
 | 
						|
  size_t index= savepoints.elements;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::last_savepoint");
 | 
						|
 | 
						|
  while (index)
 | 
						|
  {
 | 
						|
    savept= dynamic_element(&savepoints, --index, SAVEPT *);
 | 
						|
    if (savept->flags & SAVEPOINT_REALIZED)
 | 
						|
    break;
 | 
						|
  savept= NULL;
 | 
						|
  }
 | 
						|
 | 
						|
  DBUG_RETURN(savept ? savept->level : 0);
 | 
						|
}
 | 
						|
 | 
						|
bool federatedx_io_mysql::is_autocommit() const
 | 
						|
{
 | 
						|
  return actual_autocommit;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int federatedx_io_mysql::savepoint_set(ulong sp)
 | 
						|
{
 | 
						|
  int error;
 | 
						|
  SAVEPT savept;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::savepoint_set");
 | 
						|
  DBUG_PRINT("info",("savepoint=%lu", sp));
 | 
						|
  DBUG_ASSERT(sp > last_savepoint());
 | 
						|
 | 
						|
  savept.level= sp;
 | 
						|
  savept.flags= 0;
 | 
						|
 | 
						|
  if ((error= insert_dynamic(&savepoints, (uchar*) &savept) ? -1 : 0))
 | 
						|
    goto err;
 | 
						|
 | 
						|
  set_active(TRUE);
 | 
						|
  mysql.reconnect= 0;
 | 
						|
  requested_autocommit= FALSE;
 | 
						|
 | 
						|
err:
 | 
						|
  DBUG_RETURN(error);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
ulong federatedx_io_mysql::savepoint_release(ulong sp)
 | 
						|
{
 | 
						|
  SAVEPT *savept, *last= NULL;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::savepoint_release");
 | 
						|
  DBUG_PRINT("info",("savepoint=%lu", sp));
 | 
						|
  
 | 
						|
  while (savepoints.elements)
 | 
						|
  {
 | 
						|
    savept= dynamic_element(&savepoints, savepoints.elements - 1, SAVEPT *);
 | 
						|
    if (savept->level < sp)
 | 
						|
      break;
 | 
						|
    if ((savept->flags & (SAVEPOINT_REALIZED | SAVEPOINT_RESTRICT)) == SAVEPOINT_REALIZED)
 | 
						|
      last= savept;
 | 
						|
    savepoints.elements--;
 | 
						|
  }
 | 
						|
 | 
						|
  if (last)
 | 
						|
  {
 | 
						|
    char buffer[STRING_BUFFER_USUAL_SIZE];
 | 
						|
    size_t length= my_snprintf(buffer, sizeof(buffer),
 | 
						|
              "RELEASE SAVEPOINT save%lu", last->level);
 | 
						|
    actual_query(buffer, length);
 | 
						|
  }
 | 
						|
 | 
						|
  DBUG_RETURN(last_savepoint()); 
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
ulong federatedx_io_mysql::savepoint_rollback(ulong sp)
 | 
						|
{
 | 
						|
  SAVEPT *savept;
 | 
						|
  size_t index;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::savepoint_release");
 | 
						|
  DBUG_PRINT("info",("savepoint=%lu", sp));
 | 
						|
  
 | 
						|
  while (savepoints.elements)
 | 
						|
  {
 | 
						|
    savept= dynamic_element(&savepoints, savepoints.elements - 1, SAVEPT *);
 | 
						|
    if (savept->level <= sp)
 | 
						|
      break;
 | 
						|
    savepoints.elements--;
 | 
						|
  }
 | 
						|
 | 
						|
  for (index= savepoints.elements, savept= NULL; index;)
 | 
						|
  {
 | 
						|
    savept= dynamic_element(&savepoints, --index, SAVEPT *);
 | 
						|
    if (savept->flags & SAVEPOINT_REALIZED)
 | 
						|
    break;
 | 
						|
  savept= NULL;
 | 
						|
  }
 | 
						|
  
 | 
						|
  if (savept && !(savept->flags & SAVEPOINT_RESTRICT))
 | 
						|
  {
 | 
						|
    char buffer[STRING_BUFFER_USUAL_SIZE];
 | 
						|
    size_t length= my_snprintf(buffer, sizeof(buffer),
 | 
						|
              "ROLLBACK TO SAVEPOINT save%lu", savept->level);
 | 
						|
    actual_query(buffer, length);
 | 
						|
  }
 | 
						|
 | 
						|
  DBUG_RETURN(last_savepoint());
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void federatedx_io_mysql::savepoint_restrict(ulong sp)
 | 
						|
{
 | 
						|
  SAVEPT *savept;
 | 
						|
  size_t index= savepoints.elements;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::savepoint_restrict");
 | 
						|
  
 | 
						|
  while (index)
 | 
						|
  {
 | 
						|
    savept= dynamic_element(&savepoints, --index, SAVEPT *);
 | 
						|
  if (savept->level > sp)
 | 
						|
    continue;
 | 
						|
  if (savept->level < sp)
 | 
						|
    break;
 | 
						|
  savept->flags|= SAVEPOINT_RESTRICT;
 | 
						|
  break;
 | 
						|
  }
 | 
						|
  
 | 
						|
  DBUG_VOID_RETURN;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int federatedx_io_mysql::simple_query(const char *fmt, ...)
 | 
						|
{
 | 
						|
  char buffer[STRING_BUFFER_USUAL_SIZE];
 | 
						|
  size_t length;
 | 
						|
  int error;
 | 
						|
  va_list arg;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::simple_query");
 | 
						|
 | 
						|
  va_start(arg, fmt);  
 | 
						|
  length= my_vsnprintf(buffer, sizeof(buffer), fmt, arg);
 | 
						|
  va_end(arg);
 | 
						|
  
 | 
						|
  error= query(buffer, length);
 | 
						|
  
 | 
						|
  DBUG_RETURN(error);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool federatedx_io_mysql::test_all_restrict() const
 | 
						|
{
 | 
						|
  bool result= FALSE;
 | 
						|
  SAVEPT *savept;
 | 
						|
  size_t index= savepoints.elements;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::test_all_restrict");
 | 
						|
  
 | 
						|
  while (index)
 | 
						|
  {
 | 
						|
    savept= dynamic_element(&savepoints, --index, SAVEPT *);
 | 
						|
  if ((savept->flags & (SAVEPOINT_REALIZED | 
 | 
						|
                        SAVEPOINT_RESTRICT)) == SAVEPOINT_REALIZED ||
 | 
						|
    (savept->flags & SAVEPOINT_EMITTED))
 | 
						|
      DBUG_RETURN(FALSE);
 | 
						|
    if (savept->flags & SAVEPOINT_RESTRICT)
 | 
						|
    result= TRUE;
 | 
						|
  }
 | 
						|
  
 | 
						|
  DBUG_RETURN(result); 
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int federatedx_io_mysql::query(const char *buffer, size_t length)
 | 
						|
{
 | 
						|
  int error;
 | 
						|
  bool wants_autocommit= requested_autocommit | is_readonly();
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::query");
 | 
						|
 | 
						|
  if (!wants_autocommit && test_all_restrict())
 | 
						|
    wants_autocommit= TRUE;
 | 
						|
 | 
						|
  if (wants_autocommit != actual_autocommit)
 | 
						|
  {
 | 
						|
    if ((error= actual_query(wants_autocommit ? "SET AUTOCOMMIT=1"
 | 
						|
                                            : "SET AUTOCOMMIT=0", 16)))
 | 
						|
    DBUG_RETURN(error);                         
 | 
						|
    mysql.reconnect= wants_autocommit ? 1 : 0;
 | 
						|
    actual_autocommit= wants_autocommit;
 | 
						|
  }
 | 
						|
  
 | 
						|
  if (!actual_autocommit && last_savepoint() != actual_savepoint())
 | 
						|
  {
 | 
						|
    SAVEPT *savept= dynamic_element(&savepoints, savepoints.elements - 1, 
 | 
						|
                                SAVEPT *);
 | 
						|
    if (!(savept->flags & SAVEPOINT_RESTRICT))
 | 
						|
  {
 | 
						|
      char buf[STRING_BUFFER_USUAL_SIZE];
 | 
						|
      size_t len= my_snprintf(buf, sizeof(buf),
 | 
						|
                  "SAVEPOINT save%lu", savept->level);
 | 
						|
      if ((error= actual_query(buf, len)))
 | 
						|
    DBUG_RETURN(error);                         
 | 
						|
    set_active(TRUE);
 | 
						|
    savept->flags|= SAVEPOINT_EMITTED;
 | 
						|
    }
 | 
						|
    savept->flags|= SAVEPOINT_REALIZED;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!(error= actual_query(buffer, length)))
 | 
						|
    set_active(is_active() || !actual_autocommit);
 | 
						|
 | 
						|
  DBUG_RETURN(error);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int federatedx_io_mysql::actual_query(const char *buffer, size_t length)
 | 
						|
{
 | 
						|
  int error;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::actual_query");
 | 
						|
 | 
						|
  if (!mysql.net.vio)
 | 
						|
  {
 | 
						|
    my_bool my_true= 1;
 | 
						|
    my_bool my_false= 0;
 | 
						|
 | 
						|
    if (!(mysql_init(&mysql)))
 | 
						|
      DBUG_RETURN(-1);
 | 
						|
  
 | 
						|
    /*
 | 
						|
	BUG# 17044 Federated Storage Engine is not UTF8 clean
 | 
						|
	Add set names to whatever charset the table is at open
 | 
						|
	of table
 | 
						|
    */
 | 
						|
    /* this sets the csname like 'set names utf8' */
 | 
						|
    mysql_options(&mysql, MYSQL_SET_CHARSET_NAME, get_charsetname());
 | 
						|
    mysql_options(&mysql, MYSQL_OPT_USE_THREAD_SPECIFIC_MEMORY, &my_true);
 | 
						|
    mysql_options(&mysql, MYSQL_OPT_SSL_VERIFY_SERVER_CERT, &my_false);
 | 
						|
 | 
						|
    if (!mysql_real_connect(&mysql, get_hostname(), get_username(),
 | 
						|
                            get_password(), get_database(), get_port(),
 | 
						|
                            get_socket(), 0))
 | 
						|
      DBUG_RETURN(ER_CONNECT_TO_FOREIGN_DATA_SOURCE);
 | 
						|
 | 
						|
    if ((error= mysql_real_query(&mysql, STRING_WITH_LEN("set time_zone='+00:00'"))))
 | 
						|
      DBUG_RETURN(error);
 | 
						|
 | 
						|
    mysql.reconnect= 1;
 | 
						|
  }
 | 
						|
 | 
						|
  error= mysql_real_query(&mysql, buffer, (ulong)length);
 | 
						|
  
 | 
						|
  DBUG_RETURN(error);
 | 
						|
}
 | 
						|
 | 
						|
size_t federatedx_io_mysql::max_query_size() const
 | 
						|
{
 | 
						|
  return mysql.net.max_packet_size;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
my_ulonglong federatedx_io_mysql::affected_rows() const
 | 
						|
{
 | 
						|
  return mysql.affected_rows;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
my_ulonglong federatedx_io_mysql::last_insert_id() const
 | 
						|
{
 | 
						|
  return mysql.insert_id;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int federatedx_io_mysql::error_code()
 | 
						|
{
 | 
						|
  return mysql_errno(&mysql);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
const char *federatedx_io_mysql::error_str()
 | 
						|
{
 | 
						|
  return mysql_error(&mysql);
 | 
						|
}
 | 
						|
 | 
						|
FEDERATEDX_IO_RESULT *federatedx_io_mysql::store_result()
 | 
						|
{
 | 
						|
  FEDERATEDX_IO_RESULT *result;
 | 
						|
  DBUG_ENTER("federatedx_io_mysql::store_result");
 | 
						|
 | 
						|
  result= (FEDERATEDX_IO_RESULT *) mysql_store_result(&mysql);
 | 
						|
 | 
						|
  DBUG_RETURN(result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void federatedx_io_mysql::free_result(FEDERATEDX_IO_RESULT *io_result)
 | 
						|
{
 | 
						|
  mysql_free_result((MYSQL_RES *) io_result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
unsigned int federatedx_io_mysql::get_num_fields(FEDERATEDX_IO_RESULT *io_result)
 | 
						|
{
 | 
						|
  return mysql_num_fields((MYSQL_RES *) io_result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
my_ulonglong federatedx_io_mysql::get_num_rows(FEDERATEDX_IO_RESULT *io_result)
 | 
						|
{
 | 
						|
  return mysql_num_rows((MYSQL_RES *) io_result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
FEDERATEDX_IO_ROW *federatedx_io_mysql::fetch_row(FEDERATEDX_IO_RESULT *io_result,
 | 
						|
                                                  FEDERATEDX_IO_ROWS **current)
 | 
						|
{
 | 
						|
  MYSQL_RES *result= (MYSQL_RES*)io_result;
 | 
						|
  if (current)
 | 
						|
    *current= (FEDERATEDX_IO_ROWS *) result->data_cursor;
 | 
						|
  return (FEDERATEDX_IO_ROW *) mysql_fetch_row(result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
ulong *federatedx_io_mysql::fetch_lengths(FEDERATEDX_IO_RESULT *io_result)
 | 
						|
{
 | 
						|
  return mysql_fetch_lengths((MYSQL_RES *) io_result);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
const char *federatedx_io_mysql::get_column_data(FEDERATEDX_IO_ROW *row,
 | 
						|
                                                 unsigned int column)
 | 
						|
{
 | 
						|
  return ((MYSQL_ROW)row)[column];
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool federatedx_io_mysql::is_column_null(const FEDERATEDX_IO_ROW *row,
 | 
						|
                                         unsigned int column) const
 | 
						|
{
 | 
						|
  return !((MYSQL_ROW)row)[column];
 | 
						|
}
 | 
						|
 | 
						|
bool federatedx_io_mysql::table_metadata(ha_statistics *stats,
 | 
						|
                                         const char *table_name,
 | 
						|
                                         uint table_name_length, uint flag)
 | 
						|
{
 | 
						|
  char status_buf[FEDERATEDX_QUERY_BUFFER_SIZE];
 | 
						|
  FEDERATEDX_IO_RESULT *result= 0;
 | 
						|
  FEDERATEDX_IO_ROW *row;
 | 
						|
  String status_query_string(status_buf, sizeof(status_buf), &my_charset_bin);
 | 
						|
  int error;
 | 
						|
 | 
						|
  status_query_string.length(0);
 | 
						|
  status_query_string.append(STRING_WITH_LEN("SHOW TABLE STATUS LIKE "));
 | 
						|
  append_ident(&status_query_string, table_name,
 | 
						|
               table_name_length, value_quote_char);
 | 
						|
 | 
						|
  if (query(status_query_string.ptr(), status_query_string.length()))
 | 
						|
    goto error;
 | 
						|
 | 
						|
  status_query_string.length(0);
 | 
						|
 | 
						|
  result= store_result();
 | 
						|
 | 
						|
  /*
 | 
						|
    We're going to use fields num. 4, 12 and 13 of the resultset,
 | 
						|
    so make sure we have these fields.
 | 
						|
  */
 | 
						|
  if (!result || (get_num_fields(result) < 14))
 | 
						|
    goto error;
 | 
						|
 | 
						|
  if (!get_num_rows(result))
 | 
						|
    goto error;
 | 
						|
 | 
						|
  if (!(row= fetch_row(result)))
 | 
						|
    goto error;
 | 
						|
 | 
						|
  /*
 | 
						|
    deleted is set in ha_federatedx::info
 | 
						|
  */
 | 
						|
  /*
 | 
						|
    need to figure out what this means as far as federatedx is concerned,
 | 
						|
    since we don't have a "file"
 | 
						|
 | 
						|
    data_file_length = ?
 | 
						|
    index_file_length = ?
 | 
						|
    delete_length = ?
 | 
						|
  */
 | 
						|
  if (!is_column_null(row, 4))
 | 
						|
    stats->records= (ha_rows) my_strtoll10(get_column_data(row, 4),
 | 
						|
	                                   (char**) 0, &error);
 | 
						|
  if (!is_column_null(row, 5))
 | 
						|
    stats->mean_rec_length= (ulong) my_strtoll10(get_column_data(row, 5),
 | 
						|
	                                         (char**) 0, &error);
 | 
						|
 | 
						|
  stats->data_file_length= stats->records * stats->mean_rec_length;
 | 
						|
 | 
						|
  if (!is_column_null(row, 12))
 | 
						|
    stats->update_time= (time_t) my_strtoll10(get_column_data(row, 12),
 | 
						|
	                                      (char**) 0, &error);
 | 
						|
  if (!is_column_null(row, 13))
 | 
						|
    stats->check_time= (time_t) my_strtoll10(get_column_data(row, 13),
 | 
						|
	                                     (char**) 0, &error);
 | 
						|
 | 
						|
  free_result(result);
 | 
						|
  return 0;
 | 
						|
 | 
						|
error:
 | 
						|
  if (!mysql_errno(&mysql))
 | 
						|
  {
 | 
						|
    mysql.net.last_errno= ER_NO_SUCH_TABLE;
 | 
						|
    strmake_buf(mysql.net.last_error, "Remote table does not exist");
 | 
						|
  }
 | 
						|
  free_result(result);
 | 
						|
  return 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
size_t federatedx_io_mysql::get_ref_length() const
 | 
						|
{
 | 
						|
  return sizeof(mysql_position);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void federatedx_io_mysql::mark_position(FEDERATEDX_IO_RESULT *io_result,
 | 
						|
                                        void *ref, FEDERATEDX_IO_ROWS *current)
 | 
						|
{
 | 
						|
  mysql_position& pos= *reinterpret_cast<mysql_position*>(ref);
 | 
						|
  pos.result= (MYSQL_RES *) io_result;
 | 
						|
  pos.offset= (MYSQL_ROW_OFFSET) current;
 | 
						|
}
 | 
						|
 | 
						|
int federatedx_io_mysql::seek_position(FEDERATEDX_IO_RESULT **io_result,
 | 
						|
                                       const void *ref)
 | 
						|
{
 | 
						|
  const mysql_position& pos= *reinterpret_cast<const mysql_position*>(ref);
 | 
						|
 | 
						|
  if (!pos.result || !pos.offset)
 | 
						|
    return HA_ERR_END_OF_FILE;
 | 
						|
 | 
						|
  pos.result->current_row= 0;
 | 
						|
  pos.result->data_cursor= pos.offset;
 | 
						|
  *io_result= (FEDERATEDX_IO_RESULT*) pos.result;
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
void federatedx_io_mysql::set_thd(void *thd)
 | 
						|
{
 | 
						|
  mysql.net.thd= thd;
 | 
						|
}
 |