summaryrefslogtreecommitdiff
path: root/sql/protocol_cursor.cc
diff options
context:
space:
mode:
authorunknown <hf@deer.mysql.r18.ru>2003-04-23 19:37:33 +0500
committerunknown <hf@deer.mysql.r18.ru>2003-04-23 19:37:33 +0500
commitf0909cd71a7a1d0904845af4e5e06d213f911de1 (patch)
treeca16079f1d4f7eae6f2205e124d0c90afbec8409 /sql/protocol_cursor.cc
parent7c87a3f140ac801c0922b3a24e59381f627f105a (diff)
downloadmariadb-git-f0909cd71a7a1d0904845af4e5e06d213f911de1.tar.gz
SCRUM
Protocol_cursor class and sql-common/ directory Makefile.am: pack.c added to linked sources include/mysql.h: net_field_length_ll declaration added include/mysql_com.h: net_field_length declaration added libmysql/Makefile.am: sql-common files symlinked libmysql/Makefile.shared: pack.lo target added libmysql/libmysql.c: net_field_length removed from here sql/Makefile.am: pack.c added to the sources sql/mini_client.cc: mc_net_field_length functions replaced with net_field_length sql/protocol.h: Protocol_cursor class added
Diffstat (limited to 'sql/protocol_cursor.cc')
-rw-r--r--sql/protocol_cursor.cc143
1 files changed, 143 insertions, 0 deletions
diff --git a/sql/protocol_cursor.cc b/sql/protocol_cursor.cc
new file mode 100644
index 00000000000..19e3bb06d74
--- /dev/null
+++ b/sql/protocol_cursor.cc
@@ -0,0 +1,143 @@
+/* Copyright (C) 2000-2003 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; 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
+
+/*
+ Low level functions for storing data to be send to the MySQL client
+ The actual communction is handled by the net_xxx functions in net_serv.cc
+*/
+
+#ifdef __GNUC__
+#pragma implementation // gcc: Class implementation
+#endif
+
+#include "mysql_priv.h"
+#include <mysql.h>
+
+bool Protocol_cursor::send_fields(List<Item> *list, uint flag)
+{
+ List_iterator_fast<Item> it(*list);
+ Item *item;
+ MYSQL_FIELD *field, *client_field;
+
+ DBUG_ENTER("send_fields");
+ if (prepare_for_send(list))
+ return FALSE;
+
+ fields= (MYSQL_FIELD *)alloc_root(alloc, sizeof(MYSQL_FIELD) * field_count);
+ if (!fields)
+ goto err;
+
+ client_field= fields;
+ while ((item= it++))
+ {
+ Send_field server_field;
+ item->make_field(&server_field);
+
+ client_field->db= strdup_root(alloc, server_field.db_name);
+ client_field->table= strdup_root(alloc, server_field.table_name);
+ client_field->name= strdup_root(alloc, server_field.col_name);
+ client_field->org_table= strdup_root(alloc, server_field.org_table_name);
+ client_field->org_name= strdup_root(alloc, server_field.org_col_name);
+ client_field->length= server_field.length;
+ client_field->type= server_field.type;
+ client_field->flags= server_field.flags;
+ client_field->decimals= server_field.decimals;
+ client_field->db_length= strlen(client_field->db);
+ client_field->table_length= strlen(client_field->table);
+ client_field->name_length= strlen(client_field->name);
+ client_field->org_name_length= strlen(client_field->org_name);
+ client_field->org_table_length= strlen(client_field->org_table);
+ client_field->charsetnr= server_field.charsetnr;
+
+ if (INTERNAL_NUM_FIELD(client_field))
+ client_field->flags|= NUM_FLAG;
+
+ if (flag & 2)
+ {
+ char buff[80];
+ String tmp(buff, sizeof(buff), default_charset_info), *res;
+
+ if (!(res=item->val_str(&tmp)))
+ client_field->def= strdup_root(alloc, "");
+ else
+ client_field->def= strdup_root(alloc, tmp.ptr());
+ }
+ else
+ client_field->def=0;
+ client_field->max_length= 0;
+ ++client_field;
+ }
+
+ DBUG_RETURN(FALSE);
+ err:
+ send_error(thd, ER_OUT_OF_RESOURCES); /* purecov: inspected */
+ DBUG_RETURN(TRUE); /* purecov: inspected */
+}
+
+/* Get the length of next field. Change parameter to point at fieldstart */
+bool Protocol_cursor::write()
+{
+ byte *cp= (byte *)packet->ptr();
+ byte *end_pos= (byte *)packet->ptr() + packet->length();
+ ulong len;
+ MYSQL_FIELD *cur_field= fields;
+ MYSQL_FIELD *fields_end= fields + field_count;
+ MYSQL_ROWS *new_record;
+ byte **data;
+ byte *to;
+
+ new_record= (MYSQL_ROWS *)alloc_root(alloc,
+ sizeof(MYSQL_ROWS) + (field_count + 1)*sizeof(char *) + packet->length());
+ if (!new_record)
+ goto err;
+ data= (byte **)(new_record + 1);
+ new_record->data= (char **)data;
+
+ to= (byte *)(fields + field_count + 1);
+
+ for (; cur_field < fields_end; ++cur_field, ++data)
+ {
+ if ((len=net_field_length((uchar **)&cp)))
+ {
+ *data= 0;
+ }
+ else
+ {
+ if ((long)len > (end_pos - cp))
+ {
+// TODO error signal send_error(thd, CR_MALFORMED_PACKET);
+ return TRUE;
+ }
+ memcpy(to,(char*) cp,len);
+ to[len]=0;
+ to+=len+1;
+ cp+=len;
+ if (cur_field->max_length < len)
+ cur_field->max_length=len;
+ }
+ }
+
+ *prev_record= new_record;
+ prev_record= &new_record->next;
+ new_record->next= NULL;
+ row_count++;
+ return FALSE;
+ err:
+// TODO error signal send_error(thd, ER_OUT_OF_RESOURCES);
+ return TRUE;
+}
+
+