~drizzle-trunk/drizzle/development

520.4.14 by Monty Taylor
Removed korr.h and tztime.h from common_includes. Also removed the HAVE_DTRACE block and stuck it in autoconf.
1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
 *  vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
3
 *
4
 *  Copyright (C) 2008 Sun Microsystems
5
 *
6
 *  This program is free software; you can redistribute it and/or modify
7
 *  it under the terms of the GNU General Public License as published by
8
 *  the Free Software Foundation; version 2 of the License.
9
 *
10
 *  This program is distributed in the hope that it will be useful,
11
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
 *  GNU General Public License for more details.
14
 *
15
 *  You should have received a copy of the GNU General Public License
16
 *  along with this program; if not, write to the Free Software
17
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
18
 */
1 by brian
clean slate
19
20
21
/* Functions to handle date and time */
22
243.1.17 by Jay Pipes
FINAL PHASE removal of mysql_priv.h (Bye, bye my friend.)
23
#include <drizzled/server_includes.h>
549 by Monty Taylor
Took gettext.h out of header files.
24
#include <drizzled/error.h>
492.1.7 by Monty Taylor
Moved test() to its own file.
25
#include <drizzled/util/test.h>
520.4.14 by Monty Taylor
Removed korr.h and tztime.h from common_includes. Also removed the HAVE_DTRACE block and stuck it in autoconf.
26
#include <drizzled/tztime.h>
584.1.15 by Monty Taylor
The mega-patch from hell. Renamed sql_class to session (since that's what it is) and removed it and field and table from common_includes.
27
#include <drizzled/session.h>
520.4.14 by Monty Taylor
Removed korr.h and tztime.h from common_includes. Also removed the HAVE_DTRACE block and stuck it in autoconf.
28
29
/* Some functions to calculate dates */
1 by brian
clean slate
30
31
#ifndef TESTTIME
32
33
	/* Calc weekday from daynr */
34
	/* Returns 0 for monday, 1 for tuesday .... */
35
36
int calc_weekday(long daynr,bool sunday_first_day_of_week)
37
{
51.1.71 by Jay Pipes
Standardized TRUE/FALSE, removed/replaced DBUG symbols
38
  return ((int) ((daynr + 5L + (sunday_first_day_of_week ? 1L : 0L)) % 7));
1 by brian
clean slate
39
}
40
41
/*
42
  The bits in week_format has the following meaning:
43
   WEEK_MONDAY_FIRST (0)  If not set	Sunday is first day of week
44
      		   	  If set	Monday is first day of week
45
   WEEK_YEAR (1)	  If not set	Week is in range 0-53
46
47
   	Week 0 is returned for the the last week of the previous year (for
48
	a date at start of january) In this case one can get 53 for the
49
	first week of next year.  This flag ensures that the week is
50
	relevant for the given year. Note that this flag is only
51
	releveant if WEEK_JANUARY is not set.
52
53
			  If set	 Week is in range 1-53.
54
55
	In this case one may get week 53 for a date in January (when
56
	the week is that last week of previous year) and week 1 for a
57
	date in December.
58
59
  WEEK_FIRST_WEEKDAY (2)  If not set	Weeks are numbered according
60
			   		to ISO 8601:1988
61
			  If set	The week that contains the first
62
					'first-day-of-week' is week 1.
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
63
1 by brian
clean slate
64
	ISO 8601:1988 means that if the week containing January 1 has
65
	four or more days in the new year, then it is week 1;
66
	Otherwise it is the last week of the previous year, and the
67
	next week is week 1.
68
*/
69
482 by Brian Aker
Remove uint.
70
uint32_t calc_week(DRIZZLE_TIME *l_time, uint32_t week_behaviour, uint32_t *year)
1 by brian
clean slate
71
{
482 by Brian Aker
Remove uint.
72
  uint32_t days;
520.1.7 by Brian Aker
More ulong fixes (including a bug on row return on error)
73
  uint32_t daynr= calc_daynr(l_time->year,l_time->month,l_time->day);
74
  uint32_t first_daynr= calc_daynr(l_time->year,1,1);
1 by brian
clean slate
75
  bool monday_first= test(week_behaviour & WEEK_MONDAY_FIRST);
76
  bool week_year= test(week_behaviour & WEEK_YEAR);
77
  bool first_weekday= test(week_behaviour & WEEK_FIRST_WEEKDAY);
78
482 by Brian Aker
Remove uint.
79
  uint32_t weekday=calc_weekday(first_daynr, !monday_first);
1 by brian
clean slate
80
  *year=l_time->year;
81
82
  if (l_time->month == 1 && l_time->day <= 7-weekday)
83
  {
84
    if ((!week_year) && ((first_weekday && weekday != 0) || (!first_weekday && weekday >= 4)))
85
      return 0;
86
    week_year= 1;
87
    (*year)--;
88
    first_daynr-= (days=calc_days_in_year(*year));
89
    weekday= (weekday + 53*7- days) % 7;
90
  }
91
92
  if ((first_weekday && weekday != 0) ||
93
      (!first_weekday && weekday >= 4))
94
    days= daynr - (first_daynr+ (7-weekday));
95
  else
96
    days= daynr - (first_daynr - weekday);
97
98
  if (week_year && days >= 52*7)
99
  {
100
    weekday= (weekday + calc_days_in_year(*year)) % 7;
101
    if ((!first_weekday && weekday < 4) || (first_weekday && weekday == 0))
102
    {
103
      (*year)++;
104
      return 1;
105
    }
106
  }
107
  return days/7+1;
108
}
109
110
	/* Change a daynr to year, month and day */
111
	/* Daynr 0 is returned as date 00.00.00 */
112
482 by Brian Aker
Remove uint.
113
void get_date_from_daynr(long daynr,uint32_t *ret_year,uint32_t *ret_month,
114
			 uint32_t *ret_day)
1 by brian
clean slate
115
{
482 by Brian Aker
Remove uint.
116
  uint32_t year,temp,leap_day,day_of_year,days_in_year;
481 by Brian Aker
Remove all of uchar.
117
  unsigned char *month_pos;
1 by brian
clean slate
118
119
  if (daynr <= 365L || daynr >= 3652500)
120
  {						/* Fix if wrong daynr */
121
    *ret_year= *ret_month = *ret_day =0;
122
  }
123
  else
124
  {
895 by Brian Aker
Completion (?) of uint conversion.
125
    year= (uint32_t) (daynr*100 / 36525L);
1 by brian
clean slate
126
    temp=(((year-1)/100+1)*3)/4;
895 by Brian Aker
Completion (?) of uint conversion.
127
    day_of_year=(uint32_t) (daynr - (long) year * 365L) - (year-1)/4 +temp;
1 by brian
clean slate
128
    while (day_of_year > (days_in_year= calc_days_in_year(year)))
129
    {
130
      day_of_year-=days_in_year;
131
      (year)++;
132
    }
133
    leap_day=0;
134
    if (days_in_year == 366)
135
    {
136
      if (day_of_year > 31+28)
137
      {
138
	day_of_year--;
139
	if (day_of_year == 31+28)
140
	  leap_day=1;		/* Handle leapyears leapday */
141
      }
142
    }
143
    *ret_month=1;
144
    for (month_pos= days_in_month ;
895 by Brian Aker
Completion (?) of uint conversion.
145
	 day_of_year > (uint32_t) *month_pos ;
1 by brian
clean slate
146
	 day_of_year-= *(month_pos++), (*ret_month)++)
147
      ;
148
    *ret_year=year;
149
    *ret_day=day_of_year+leap_day;
150
  }
51.1.71 by Jay Pipes
Standardized TRUE/FALSE, removed/replaced DBUG symbols
151
  return;
1 by brian
clean slate
152
}
153
154
/*
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
155
  Convert a timestamp string to a DRIZZLE_TIME value and produce a warning
1 by brian
clean slate
156
  if string was truncated during conversion.
157
158
  NOTE
159
    See description of str_to_datetime() for more information.
160
*/
161
398.1.1 by Monty Taylor
Remove typedef enum enum_drizzle_timestamp_type timestamp_type;
162
enum enum_drizzle_timestamp_type
482 by Brian Aker
Remove uint.
163
str_to_datetime_with_warn(const char *str, uint32_t length, DRIZZLE_TIME *l_time,
164
                          uint32_t flags)
1 by brian
clean slate
165
{
166
  int was_cut;
520.1.22 by Brian Aker
Second pass of thd cleanup
167
  Session *session= current_session;
398.1.1 by Monty Taylor
Remove typedef enum enum_drizzle_timestamp_type timestamp_type;
168
  enum enum_drizzle_timestamp_type ts_type;
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
169
1 by brian
clean slate
170
  ts_type= str_to_datetime(str, length, l_time,
520.1.22 by Brian Aker
Second pass of thd cleanup
171
                           (flags | (session->variables.sql_mode &
1 by brian
clean slate
172
                                     (MODE_INVALID_DATES |
173
                                      MODE_NO_ZERO_DATE))),
174
                           &was_cut);
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
175
  if (was_cut || ts_type <= DRIZZLE_TIMESTAMP_ERROR)
520.1.22 by Brian Aker
Second pass of thd cleanup
176
    make_truncated_value_warning(current_session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
461 by Monty Taylor
Removed NullS. bu-bye.
177
                                 str, length, ts_type,  NULL);
1 by brian
clean slate
178
  return ts_type;
179
}
180
181
/*
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
182
  Convert a time string to a DRIZZLE_TIME struct and produce a warning
1 by brian
clean slate
183
  if string was cut during conversion.
184
185
  NOTE
186
    See str_to_time() for more info.
187
*/
188
bool
482 by Brian Aker
Remove uint.
189
str_to_time_with_warn(const char *str, uint32_t length, DRIZZLE_TIME *l_time)
1 by brian
clean slate
190
{
191
  int warning;
192
  bool ret_val= str_to_time(str, length, l_time, &warning);
193
  if (ret_val || warning)
520.1.22 by Brian Aker
Second pass of thd cleanup
194
    make_truncated_value_warning(current_session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
461 by Monty Taylor
Removed NullS. bu-bye.
195
                                 str, length, DRIZZLE_TIMESTAMP_TIME, NULL);
1 by brian
clean slate
196
  return ret_val;
197
}
198
199
200
/*
201
  Convert a system time structure to TIME
202
*/
203
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
204
void localtime_to_TIME(DRIZZLE_TIME *to, struct tm *from)
1 by brian
clean slate
205
{
206
  to->neg=0;
207
  to->second_part=0;
208
  to->year=	(int) ((from->tm_year+1900) % 10000);
209
  to->month=	(int) from->tm_mon+1;
210
  to->day=	(int) from->tm_mday;
211
  to->hour=	(int) from->tm_hour;
212
  to->minute=	(int) from->tm_min;
213
  to->second=   (int) from->tm_sec;
214
}
215
907.1.2 by Jay Pipes
Merging in old r902 temporal changes
216
void make_date(const DRIZZLE_TIME *l_time, String *str)
217
{
910.4.7 by Stewart Smith
ensure there is enough space allocated to String for string representation of date/time types.
218
  str->alloc(MAX_DATE_STRING_REP_LENGTH);
219
  uint32_t length= (uint32_t) my_date_to_str(l_time, str->c_ptr());
907.1.2 by Jay Pipes
Merging in old r902 temporal changes
220
  str->length(length);
221
  str->set_charset(&my_charset_bin);
222
}
223
224
225
void make_datetime(const DRIZZLE_TIME *l_time, String *str)
226
{
910.4.7 by Stewart Smith
ensure there is enough space allocated to String for string representation of date/time types.
227
  str->alloc(MAX_DATE_STRING_REP_LENGTH);
228
  uint32_t length= (uint32_t) my_datetime_to_str(l_time, str->c_ptr());
1 by brian
clean slate
229
  str->length(length);
230
  str->set_charset(&my_charset_bin);
231
}
232
233
520.1.22 by Brian Aker
Second pass of thd cleanup
234
void make_truncated_value_warning(Session *session, DRIZZLE_ERROR::enum_warning_level level,
1 by brian
clean slate
235
                                  const char *str_val,
482 by Brian Aker
Remove uint.
236
				  uint32_t str_length,
398.1.1 by Monty Taylor
Remove typedef enum enum_drizzle_timestamp_type timestamp_type;
237
                                  enum enum_drizzle_timestamp_type time_type,
1 by brian
clean slate
238
                                  const char *field_name)
239
{
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
240
  char warn_buff[DRIZZLE_ERRMSG_SIZE];
1 by brian
clean slate
241
  const char *type_str;
383.1.12 by Brian Aker
Much closer toward UTF8 being around all the time...
242
  CHARSET_INFO *cs= &my_charset_utf8_general_ci;
1 by brian
clean slate
243
  char buff[128];
205 by Brian Aker
uint32 -> uin32_t
244
  String str(buff,(uint32_t) sizeof(buff), system_charset_info);
1 by brian
clean slate
245
  str.copy(str_val, str_length, system_charset_info);
246
  str[str_length]= 0;               // Ensure we have end 0 for snprintf
247
248
  switch (time_type) {
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
249
    case DRIZZLE_TIMESTAMP_DATE:
1 by brian
clean slate
250
      type_str= "date";
251
      break;
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
252
    case DRIZZLE_TIMESTAMP_TIME:
1 by brian
clean slate
253
      type_str= "time";
254
      break;
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
255
    case DRIZZLE_TIMESTAMP_DATETIME:  // FALLTHROUGH
1 by brian
clean slate
256
    default:
257
      type_str= "datetime";
258
      break;
259
  }
260
  if (field_name)
261
    cs->cset->snprintf(cs, warn_buff, sizeof(warn_buff),
262
                       ER(ER_TRUNCATED_WRONG_VALUE_FOR_FIELD),
263
                       type_str, str.c_ptr(), field_name,
520.1.22 by Brian Aker
Second pass of thd cleanup
264
                       (uint32_t) session->row_count);
1 by brian
clean slate
265
  else
266
  {
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
267
    if (time_type > DRIZZLE_TIMESTAMP_ERROR)
1 by brian
clean slate
268
      cs->cset->snprintf(cs, warn_buff, sizeof(warn_buff),
269
                         ER(ER_TRUNCATED_WRONG_VALUE),
270
                         type_str, str.c_ptr());
271
    else
272
      cs->cset->snprintf(cs, warn_buff, sizeof(warn_buff),
273
                         ER(ER_WRONG_VALUE), type_str, str.c_ptr());
274
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
275
  push_warning(session, level,
1 by brian
clean slate
276
               ER_TRUNCATED_WRONG_VALUE, warn_buff);
277
}
278
279
/*
280
  Calculate difference between two datetime values as seconds + microseconds.
281
282
  SYNOPSIS
283
    calc_time_diff()
284
      l_time1         - TIME/DATE/DATETIME value
285
      l_time2         - TIME/DATE/DATETIME value
286
      l_sign          - 1 absolute values are substracted,
287
                        -1 absolute values are added.
288
      seconds_out     - Out parameter where difference between
289
                        l_time1 and l_time2 in seconds is stored.
290
      microseconds_out- Out parameter where microsecond part of difference
291
                        between l_time1 and l_time2 is stored.
292
293
  NOTE
294
    This function calculates difference between l_time1 and l_time2 absolute
295
    values. So one should set l_sign and correct result if he want to take
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
296
    signs into account (i.e. for DRIZZLE_TIME values).
1 by brian
clean slate
297
298
  RETURN VALUES
299
    Returns sign of difference.
300
    1 means negative result
301
    0 means positive result
302
303
*/
304
305
bool
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
306
calc_time_diff(DRIZZLE_TIME *l_time1, DRIZZLE_TIME *l_time2, int l_sign, int64_t *seconds_out,
1 by brian
clean slate
307
               long *microseconds_out)
308
{
309
  long days;
310
  bool neg;
152 by Brian Aker
longlong replacement
311
  int64_t microseconds;
1 by brian
clean slate
312
313
  /*
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
314
    We suppose that if first argument is DRIZZLE_TIMESTAMP_TIME
1 by brian
clean slate
315
    the second argument should be TIMESTAMP_TIME also.
316
    We should check it before calc_time_diff call.
317
  */
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
318
  if (l_time1->time_type == DRIZZLE_TIMESTAMP_TIME)  // Time value
1 by brian
clean slate
319
    days= (long)l_time1->day - l_sign * (long)l_time2->day;
320
  else
321
  {
895 by Brian Aker
Completion (?) of uint conversion.
322
    days= calc_daynr((uint32_t) l_time1->year,
323
		     (uint32_t) l_time1->month,
324
		     (uint32_t) l_time1->day);
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
325
    if (l_time2->time_type == DRIZZLE_TIMESTAMP_TIME)
1 by brian
clean slate
326
      days-= l_sign * (long)l_time2->day;
327
    else
895 by Brian Aker
Completion (?) of uint conversion.
328
      days-= l_sign*calc_daynr((uint32_t) l_time2->year,
329
			       (uint32_t) l_time2->month,
330
			       (uint32_t) l_time2->day);
1 by brian
clean slate
331
  }
332
398.1.8 by Monty Taylor
Enabled -Wlong-long.
333
  microseconds= ((int64_t)days*86400L +
152 by Brian Aker
longlong replacement
334
                 (int64_t)(l_time1->hour*3600L +
1 by brian
clean slate
335
                            l_time1->minute*60L +
336
                            l_time1->second) -
152 by Brian Aker
longlong replacement
337
                 l_sign*(int64_t)(l_time2->hour*3600L +
1 by brian
clean slate
338
                                   l_time2->minute*60L +
398.1.8 by Monty Taylor
Enabled -Wlong-long.
339
                                   l_time2->second)) * 1000000L +
152 by Brian Aker
longlong replacement
340
                (int64_t)l_time1->second_part -
341
                l_sign*(int64_t)l_time2->second_part;
1 by brian
clean slate
342
343
  neg= 0;
344
  if (microseconds < 0)
345
  {
346
    microseconds= -microseconds;
347
    neg= 1;
348
  }
349
  *seconds_out= microseconds/1000000L;
350
  *microseconds_out= (long) (microseconds%1000000L);
351
  return neg;
352
}
353
354
355
#endif