mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-31 15:50:51 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			217 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			217 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* Copyright (C) 2006 MySQL AB
 | |
| 
 | |
|    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; version 2 of the License.
 | |
| 
 | |
|    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 St, Fifth Floor, Boston, MA  02110-1301  USA */
 | |
| 
 | |
| #include "rpl_utility.h"
 | |
| #include "rpl_rli.h"
 | |
| 
 | |
| /*********************************************************************
 | |
|  *                   table_def member definitions                    *
 | |
|  *********************************************************************/
 | |
| 
 | |
| /*
 | |
|   This function returns the field size in raw bytes based on the type
 | |
|   and the encoded field data from the master's raw data.
 | |
| */
 | |
| uint32 table_def::calc_field_size(uint col, uchar *master_data) const
 | |
| {
 | |
|   uint32 length;
 | |
| 
 | |
|   switch (type(col)) {
 | |
|   case MYSQL_TYPE_NEWDECIMAL:
 | |
|     length= my_decimal_get_binary_size(m_field_metadata[col] >> 8, 
 | |
|              m_field_metadata[col] - ((m_field_metadata[col] >> 8) << 8));
 | |
|     break;
 | |
|   case MYSQL_TYPE_DECIMAL:
 | |
|   case MYSQL_TYPE_FLOAT:
 | |
|   case MYSQL_TYPE_DOUBLE:
 | |
|     length= m_field_metadata[col];
 | |
|     break;
 | |
|   case MYSQL_TYPE_SET:
 | |
|   case MYSQL_TYPE_ENUM:
 | |
|   case MYSQL_TYPE_STRING:
 | |
|   {
 | |
|     if (((m_field_metadata[col] & 0xff00) == (MYSQL_TYPE_SET << 8)) ||
 | |
|         ((m_field_metadata[col] & 0xff00) == (MYSQL_TYPE_ENUM << 8)))
 | |
|       length= m_field_metadata[col] & 0x00ff;
 | |
|     else
 | |
|     {
 | |
|       length= m_field_metadata[col] & 0x00ff;
 | |
|       DBUG_ASSERT(length > 0);
 | |
|       if (length > 255)
 | |
|       {
 | |
|         DBUG_ASSERT(uint2korr(master_data) > 0);
 | |
|         length= uint2korr(master_data) + 2;
 | |
|       }
 | |
|       else
 | |
|         length= (uint) *master_data + 1;
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   case MYSQL_TYPE_YEAR:
 | |
|   case MYSQL_TYPE_TINY:
 | |
|     length= 1;
 | |
|     break;
 | |
|   case MYSQL_TYPE_SHORT:
 | |
|     length= 2;
 | |
|     break;
 | |
|   case MYSQL_TYPE_INT24:
 | |
|     length= 3;
 | |
|     break;
 | |
|   case MYSQL_TYPE_LONG:
 | |
|     length= 4;
 | |
|     break;
 | |
| #ifdef HAVE_LONG_LONG
 | |
|   case MYSQL_TYPE_LONGLONG:
 | |
|     length= 8;
 | |
|     break;
 | |
| #endif
 | |
|   case MYSQL_TYPE_NULL:
 | |
|     length= 0;
 | |
|     break;
 | |
|   case MYSQL_TYPE_NEWDATE:
 | |
|     length= 3;
 | |
|     break;
 | |
|   case MYSQL_TYPE_DATE:
 | |
|   case MYSQL_TYPE_TIME:
 | |
|     length= 3;
 | |
|     break;
 | |
|   case MYSQL_TYPE_TIMESTAMP:
 | |
|     length= 4;
 | |
|     break;
 | |
|   case MYSQL_TYPE_DATETIME:
 | |
|     length= 8;
 | |
|     break;
 | |
|   case MYSQL_TYPE_BIT:
 | |
|   {
 | |
|     uint from_len= (m_field_metadata[col] >> 8U) & 0x00ff;
 | |
|     uint from_bit_len= m_field_metadata[col] & 0x00ff;
 | |
|     DBUG_ASSERT(from_bit_len <= 7);
 | |
|     length= from_len + ((from_bit_len > 0) ? 1 : 0);
 | |
|     break;
 | |
|   }
 | |
|   case MYSQL_TYPE_VARCHAR:
 | |
|   {
 | |
|     length= m_field_metadata[col] > 255 ? 2 : 1; // c&p of Field_varstring::data_length()
 | |
|     DBUG_ASSERT(uint2korr(master_data) > 0);
 | |
|     length+= length == 1 ? (uint32) *master_data : uint2korr(master_data);
 | |
|     break;
 | |
|   }
 | |
|   case MYSQL_TYPE_TINY_BLOB:
 | |
|   case MYSQL_TYPE_MEDIUM_BLOB:
 | |
|   case MYSQL_TYPE_LONG_BLOB:
 | |
|   case MYSQL_TYPE_BLOB:
 | |
|   case MYSQL_TYPE_GEOMETRY:
 | |
|   {
 | |
| #if 1
 | |
|     /*
 | |
|       BUG#29549: 
 | |
|       This is currently broken for NDB, which is using big-endian
 | |
|       order when packing length of BLOB. Once they have decided how to
 | |
|       fix the issue, we can enable the code below to make sure to
 | |
|       always read the length in little-endian order.
 | |
|     */
 | |
|     Field_blob fb(m_field_metadata[col]);
 | |
|     length= fb.get_packed_size(master_data, TRUE);
 | |
| #else
 | |
|     /*
 | |
|       Compute the length of the data. We cannot use get_length() here
 | |
|       since it is dependent on the specific table (and also checks the
 | |
|       packlength using the internal 'table' pointer) and replication
 | |
|       is using a fixed format for storing data in the binlog.
 | |
|     */
 | |
|     switch (m_field_metadata[col]) {
 | |
|     case 1:
 | |
|       length= *master_data;
 | |
|       break;
 | |
|     case 2:
 | |
|       length= sint2korr(master_data);
 | |
|       break;
 | |
|     case 3:
 | |
|       length= uint3korr(master_data);
 | |
|       break;
 | |
|     case 4:
 | |
|       length= uint4korr(master_data);
 | |
|       break;
 | |
|     default:
 | |
|       DBUG_ASSERT(0);		// Should not come here
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     length+= m_field_metadata[col];
 | |
| #endif
 | |
|     break;
 | |
|   }
 | |
|   default:
 | |
|     length= -1;
 | |
|   }
 | |
|   return length;
 | |
| }
 | |
| 
 | |
| /*
 | |
|   Is the definition compatible with a table?
 | |
| 
 | |
| */
 | |
| int
 | |
| table_def::compatible_with(Relay_log_info const *rli_arg, TABLE *table)
 | |
|   const
 | |
| {
 | |
|   /*
 | |
|     We only check the initial columns for the tables.
 | |
|   */
 | |
|   uint const cols_to_check= min(table->s->fields, size());
 | |
|   int error= 0;
 | |
|   Relay_log_info const *rli= const_cast<Relay_log_info*>(rli_arg);
 | |
| 
 | |
|   TABLE_SHARE const *const tsh= table->s;
 | |
| 
 | |
|   for (uint col= 0 ; col < cols_to_check ; ++col)
 | |
|   {
 | |
|     if (table->field[col]->type() != type(col))
 | |
|     {
 | |
|       DBUG_ASSERT(col < size() && col < tsh->fields);
 | |
|       DBUG_ASSERT(tsh->db.str && tsh->table_name.str);
 | |
|       error= 1;
 | |
|       char buf[256];
 | |
|       my_snprintf(buf, sizeof(buf), "Column %d type mismatch - "
 | |
|                   "received type %d, %s.%s has type %d",
 | |
|                   col, type(col), tsh->db.str, tsh->table_name.str,
 | |
|                   table->field[col]->type());
 | |
|       rli->report(ERROR_LEVEL, ER_BINLOG_ROW_WRONG_TABLE_DEF,
 | |
|                   ER(ER_BINLOG_ROW_WRONG_TABLE_DEF), buf);
 | |
|     }
 | |
|     /*
 | |
|       Check the slave's field size against that of the master.
 | |
|     */
 | |
|     if (!error && 
 | |
|         !table->field[col]->compatible_field_size(field_metadata(col)))
 | |
|     {
 | |
|       error= 1;
 | |
|       char buf[256];
 | |
|       my_snprintf(buf, sizeof(buf), "Column %d size mismatch - "
 | |
|                   "master has size %d, %s.%s on slave has size %d."
 | |
|                   " Master's column size should be <= the slave's "
 | |
|                   "column size.", col,
 | |
|                   table->field[col]->pack_length_from_metadata(
 | |
|                                        m_field_metadata[col]),
 | |
|                   tsh->db.str, tsh->table_name.str, 
 | |
|                   table->field[col]->row_pack_length());
 | |
|       rli->report(ERROR_LEVEL, ER_BINLOG_ROW_WRONG_TABLE_DEF,
 | |
|                   ER(ER_BINLOG_ROW_WRONG_TABLE_DEF), buf);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return error;
 | |
| }
 | 
