misc.c
changeset 4261 28670f743746
parent 3977 513433ebd092
child 4285 19cb745c694e
equal deleted inserted replaced
4260:c638c6441121 4261:28670f743746
    10 #include "table/strings.h"
    10 #include "table/strings.h"
    11 #include "table/sprites.h"
    11 #include "table/sprites.h"
    12 #include "map.h"
    12 #include "map.h"
    13 #include "vehicle.h"
    13 #include "vehicle.h"
    14 #include "saveload.h"
    14 #include "saveload.h"
    15 #include "network.h"
       
    16 #include "network_data.h"
       
    17 #include "network_server.h"
       
    18 #include "engine.h"
    15 #include "engine.h"
    19 #include "vehicle_gui.h"
    16 #include "vehicle_gui.h"
    20 #include "variables.h"
    17 #include "variables.h"
    21 #include "ai/ai.h"
    18 #include "ai/ai.h"
    22 #include "table/landscape_const.h"
    19 #include "table/landscape_const.h"
       
    20 #include "date.h"
    23 
    21 
    24 extern void StartupEconomy(void);
    22 extern void StartupEconomy(void);
    25 
    23 
    26 char _name_array[512][32];
    24 char _name_array[512][32];
    27 
    25 
    72 }
    70 }
    73 
    71 
    74 uint InteractiveRandomRange(uint max)
    72 uint InteractiveRandomRange(uint max)
    75 {
    73 {
    76 	return GB(InteractiveRandom(), 0, 16) * max >> 16;
    74 	return GB(InteractiveRandom(), 0, 16) * max >> 16;
    77 }
       
    78 
       
    79 void SetDate(uint date)
       
    80 {
       
    81 	YearMonthDay ymd;
       
    82 
       
    83 	_date = date;
       
    84 	ConvertDayToYMD(&ymd, date);
       
    85 	_cur_year = ymd.year;
       
    86 	_cur_month = ymd.month;
       
    87 #ifdef ENABLE_NETWORK
       
    88 	_network_last_advertise_frame = 0;
       
    89 	_network_need_advertise = true;
       
    90 #endif /* ENABLE_NETWORK */
       
    91 }
    75 }
    92 
    76 
    93 void InitializeVehicles(void);
    77 void InitializeVehicles(void);
    94 void InitializeWaypoints(void);
    78 void InitializeWaypoints(void);
    95 void InitializeDepot(void);
    79 void InitializeDepot(void);
   275 		_error_message = STR_0131_TOO_MANY_NAMES_DEFINED;
   259 		_error_message = STR_0131_TOO_MANY_NAMES_DEFINED;
   276 		return 0;
   260 		return 0;
   277 	}
   261 	}
   278 }
   262 }
   279 
   263 
   280 
       
   281 #define M(a,b) ((a<<5)|b)
       
   282 static const uint16 _month_date_from_year_day[] = {
       
   283 M(0,1),M(0,2),M(0,3),M(0,4),M(0,5),M(0,6),M(0,7),M(0,8),M(0,9),M(0,10),M(0,11),M(0,12),M(0,13),M(0,14),M(0,15),M(0,16),M(0,17),M(0,18),M(0,19),M(0,20),M(0,21),M(0,22),M(0,23),M(0,24),M(0,25),M(0,26),M(0,27),M(0,28),M(0,29),M(0,30),M(0,31),
       
   284 M(1,1),M(1,2),M(1,3),M(1,4),M(1,5),M(1,6),M(1,7),M(1,8),M(1,9),M(1,10),M(1,11),M(1,12),M(1,13),M(1,14),M(1,15),M(1,16),M(1,17),M(1,18),M(1,19),M(1,20),M(1,21),M(1,22),M(1,23),M(1,24),M(1,25),M(1,26),M(1,27),M(1,28),M(1,29),
       
   285 M(2,1),M(2,2),M(2,3),M(2,4),M(2,5),M(2,6),M(2,7),M(2,8),M(2,9),M(2,10),M(2,11),M(2,12),M(2,13),M(2,14),M(2,15),M(2,16),M(2,17),M(2,18),M(2,19),M(2,20),M(2,21),M(2,22),M(2,23),M(2,24),M(2,25),M(2,26),M(2,27),M(2,28),M(2,29),M(2,30),M(2,31),
       
   286 M(3,1),M(3,2),M(3,3),M(3,4),M(3,5),M(3,6),M(3,7),M(3,8),M(3,9),M(3,10),M(3,11),M(3,12),M(3,13),M(3,14),M(3,15),M(3,16),M(3,17),M(3,18),M(3,19),M(3,20),M(3,21),M(3,22),M(3,23),M(3,24),M(3,25),M(3,26),M(3,27),M(3,28),M(3,29),M(3,30),
       
   287 M(4,1),M(4,2),M(4,3),M(4,4),M(4,5),M(4,6),M(4,7),M(4,8),M(4,9),M(4,10),M(4,11),M(4,12),M(4,13),M(4,14),M(4,15),M(4,16),M(4,17),M(4,18),M(4,19),M(4,20),M(4,21),M(4,22),M(4,23),M(4,24),M(4,25),M(4,26),M(4,27),M(4,28),M(4,29),M(4,30),M(4,31),
       
   288 M(5,1),M(5,2),M(5,3),M(5,4),M(5,5),M(5,6),M(5,7),M(5,8),M(5,9),M(5,10),M(5,11),M(5,12),M(5,13),M(5,14),M(5,15),M(5,16),M(5,17),M(5,18),M(5,19),M(5,20),M(5,21),M(5,22),M(5,23),M(5,24),M(5,25),M(5,26),M(5,27),M(5,28),M(5,29),M(5,30),
       
   289 M(6,1),M(6,2),M(6,3),M(6,4),M(6,5),M(6,6),M(6,7),M(6,8),M(6,9),M(6,10),M(6,11),M(6,12),M(6,13),M(6,14),M(6,15),M(6,16),M(6,17),M(6,18),M(6,19),M(6,20),M(6,21),M(6,22),M(6,23),M(6,24),M(6,25),M(6,26),M(6,27),M(6,28),M(6,29),M(6,30),M(6,31),
       
   290 M(7,1),M(7,2),M(7,3),M(7,4),M(7,5),M(7,6),M(7,7),M(7,8),M(7,9),M(7,10),M(7,11),M(7,12),M(7,13),M(7,14),M(7,15),M(7,16),M(7,17),M(7,18),M(7,19),M(7,20),M(7,21),M(7,22),M(7,23),M(7,24),M(7,25),M(7,26),M(7,27),M(7,28),M(7,29),M(7,30),M(7,31),
       
   291 M(8,1),M(8,2),M(8,3),M(8,4),M(8,5),M(8,6),M(8,7),M(8,8),M(8,9),M(8,10),M(8,11),M(8,12),M(8,13),M(8,14),M(8,15),M(8,16),M(8,17),M(8,18),M(8,19),M(8,20),M(8,21),M(8,22),M(8,23),M(8,24),M(8,25),M(8,26),M(8,27),M(8,28),M(8,29),M(8,30),
       
   292 M(9,1),M(9,2),M(9,3),M(9,4),M(9,5),M(9,6),M(9,7),M(9,8),M(9,9),M(9,10),M(9,11),M(9,12),M(9,13),M(9,14),M(9,15),M(9,16),M(9,17),M(9,18),M(9,19),M(9,20),M(9,21),M(9,22),M(9,23),M(9,24),M(9,25),M(9,26),M(9,27),M(9,28),M(9,29),M(9,30),M(9,31),
       
   293 M(10,1),M(10,2),M(10,3),M(10,4),M(10,5),M(10,6),M(10,7),M(10,8),M(10,9),M(10,10),M(10,11),M(10,12),M(10,13),M(10,14),M(10,15),M(10,16),M(10,17),M(10,18),M(10,19),M(10,20),M(10,21),M(10,22),M(10,23),M(10,24),M(10,25),M(10,26),M(10,27),M(10,28),M(10,29),M(10,30),
       
   294 M(11,1),M(11,2),M(11,3),M(11,4),M(11,5),M(11,6),M(11,7),M(11,8),M(11,9),M(11,10),M(11,11),M(11,12),M(11,13),M(11,14),M(11,15),M(11,16),M(11,17),M(11,18),M(11,19),M(11,20),M(11,21),M(11,22),M(11,23),M(11,24),M(11,25),M(11,26),M(11,27),M(11,28),M(11,29),M(11,30),M(11,31),
       
   295 };
       
   296 #undef M
       
   297 
       
   298 enum {
       
   299 	ACCUM_JAN = 0,
       
   300 	ACCUM_FEB = ACCUM_JAN + 31,
       
   301 	ACCUM_MAR = ACCUM_FEB + 29,
       
   302 	ACCUM_APR = ACCUM_MAR + 31,
       
   303 	ACCUM_MAY = ACCUM_APR + 30,
       
   304 	ACCUM_JUN = ACCUM_MAY + 31,
       
   305 	ACCUM_JUL = ACCUM_JUN + 30,
       
   306 	ACCUM_AUG = ACCUM_JUL + 31,
       
   307 	ACCUM_SEP = ACCUM_AUG + 31,
       
   308 	ACCUM_OCT = ACCUM_SEP + 30,
       
   309 	ACCUM_NOV = ACCUM_OCT + 31,
       
   310 	ACCUM_DEC = ACCUM_NOV + 30,
       
   311 };
       
   312 
       
   313 static const uint16 _accum_days_for_month[] = {
       
   314 	ACCUM_JAN,ACCUM_FEB,ACCUM_MAR,ACCUM_APR,
       
   315 	ACCUM_MAY,ACCUM_JUN,ACCUM_JUL,ACCUM_AUG,
       
   316 	ACCUM_SEP,ACCUM_OCT,ACCUM_NOV,ACCUM_DEC,
       
   317 };
       
   318 
       
   319 
       
   320 void ConvertDayToYMD(YearMonthDay *ymd, uint16 date)
       
   321 {
       
   322 	uint yr = date / (365+365+365+366);
       
   323 	uint rem = date % (365+365+365+366);
       
   324 	uint x;
       
   325 
       
   326 	yr *= 4;
       
   327 
       
   328 	if (rem >= 366) {
       
   329 		rem--;
       
   330 		do {
       
   331 			rem -= 365;
       
   332 			yr++;
       
   333 		} while (rem >= 365);
       
   334 		if (rem >= 31+28) rem++;
       
   335 	}
       
   336 
       
   337 	ymd->year = yr;
       
   338 
       
   339 	x = _month_date_from_year_day[rem];
       
   340 	ymd->month = x >> 5;
       
   341 	ymd->day = x & 0x1F;
       
   342 }
       
   343 
       
   344 // year is a number between 0..?
       
   345 // month is a number between 0..11
       
   346 // day is a number between 1..31
       
   347 uint ConvertYMDToDay(uint year, uint month, uint day)
       
   348 {
       
   349 	uint rem;
       
   350 
       
   351 	// day in the year
       
   352 	rem = _accum_days_for_month[month] + day - 1;
       
   353 
       
   354 	// remove feb 29 from year 1,2,3
       
   355 	if (year & 3) rem += (year & 3) * 365 + (rem < 31+29);
       
   356 
       
   357 	// base date.
       
   358 	return (year >> 2) * (365+365+365+366) + rem;
       
   359 }
       
   360 
       
   361 // convert a date on the form
       
   362 // 1920 - 2090 (MAX_YEAR_END_REAL)
       
   363 // 192001 - 209012
       
   364 // 19200101 - 20901231
       
   365 // or if > 2090 and below 65536, treat it as a daycount
       
   366 // returns -1 if no conversion was possible
       
   367 uint ConvertIntDate(uint date)
       
   368 {
       
   369 	uint year, month = 0, day = 1;
       
   370 
       
   371 	if (IS_INT_INSIDE(date, 1920, MAX_YEAR_END_REAL + 1)) {
       
   372 		year = date - 1920;
       
   373 	} else if (IS_INT_INSIDE(date, 192001, 209012+1)) {
       
   374 		month = date % 100 - 1;
       
   375 		year = date / 100 - 1920;
       
   376 	} else if (IS_INT_INSIDE(date, 19200101, 20901231+1)) {
       
   377 		day = date % 100; date /= 100;
       
   378 		month = date % 100 - 1;
       
   379 		year = date / 100 - 1920;
       
   380 	} else if (IS_INT_INSIDE(date, 2091, 65536)) {
       
   381 		return date;
       
   382 	} else {
       
   383 		return (uint)-1;
       
   384 	}
       
   385 
       
   386 	// invalid ranges?
       
   387 	if (month >= 12 || !IS_INT_INSIDE(day, 1, 31+1)) return (uint)-1;
       
   388 
       
   389 	return ConvertYMDToDay(year, month, day);
       
   390 }
       
   391 
       
   392 
       
   393 // Calculate constants that depend on the landscape type.
   264 // Calculate constants that depend on the landscape type.
   394 void InitializeLandscapeVariables(bool only_constants)
   265 void InitializeLandscapeVariables(bool only_constants)
   395 {
   266 {
   396 	const CargoTypesValues *lpd;
   267 	const CargoTypesValues *lpd;
   397 	uint i;
   268 	uint i;
   420 		_cargoc.transit_days_2[i] = lpd->transit_days_table_2[i];
   291 		_cargoc.transit_days_2[i] = lpd->transit_days_table_2[i];
   421 	}
   292 	}
   422 }
   293 }
   423 
   294 
   424 
   295 
   425 void OnNewDay_Train(Vehicle *v);
       
   426 void OnNewDay_RoadVeh(Vehicle *v);
       
   427 void OnNewDay_Aircraft(Vehicle *v);
       
   428 void OnNewDay_Ship(Vehicle *v);
       
   429 static void OnNewDay_EffectVehicle(Vehicle *v) { /* empty */ }
       
   430 void OnNewDay_DisasterVehicle(Vehicle *v);
       
   431 
       
   432 typedef void OnNewVehicleDayProc(Vehicle *v);
       
   433 
       
   434 static OnNewVehicleDayProc * _on_new_vehicle_day_proc[] = {
       
   435 	OnNewDay_Train,
       
   436 	OnNewDay_RoadVeh,
       
   437 	OnNewDay_Ship,
       
   438 	OnNewDay_Aircraft,
       
   439 	OnNewDay_EffectVehicle,
       
   440 	OnNewDay_DisasterVehicle,
       
   441 };
       
   442 
       
   443 void EnginesDailyLoop(void);
       
   444 void DisasterDailyLoop(void);
       
   445 void PlayersMonthlyLoop(void);
       
   446 void EnginesMonthlyLoop(void);
       
   447 void TownsMonthlyLoop(void);
       
   448 void IndustryMonthlyLoop(void);
       
   449 void StationMonthlyLoop(void);
       
   450 
       
   451 void PlayersYearlyLoop(void);
       
   452 void TrainsYearlyLoop(void);
       
   453 void RoadVehiclesYearlyLoop(void);
       
   454 void AircraftYearlyLoop(void);
       
   455 void ShipsYearlyLoop(void);
       
   456 
       
   457 void WaypointsDailyLoop(void);
       
   458 
       
   459 
       
   460 static const uint16 _autosave_months[] = {
       
   461 	0, // never
       
   462 	0xFFF, // every month
       
   463 	0x249, // every 3 months
       
   464 	0x041, // every 6 months
       
   465 	0x001, // every 12 months
       
   466 };
       
   467 
       
   468 /**
       
   469  * Runs the day_proc for every DAY_TICKS vehicle starting at daytick.
       
   470  */
       
   471 static void RunVehicleDayProc(uint daytick)
       
   472 {
       
   473 	uint total = _vehicle_pool.total_items;
       
   474 	uint i;
       
   475 
       
   476 	for (i = daytick; i < total; i += DAY_TICKS) {
       
   477 		Vehicle* v = GetVehicle(i);
       
   478 
       
   479 		if (v->type != 0) _on_new_vehicle_day_proc[v->type - 0x10](v);
       
   480 	}
       
   481 }
       
   482 
       
   483 void IncreaseDate(void)
       
   484 {
       
   485 	YearMonthDay ymd;
       
   486 
       
   487 	if (_game_mode == GM_MENU) {
       
   488 		_tick_counter++;
       
   489 		return;
       
   490 	}
       
   491 
       
   492 	RunVehicleDayProc(_date_fract);
       
   493 
       
   494 	/* increase day, and check if a new day is there? */
       
   495 	_tick_counter++;
       
   496 
       
   497 	_date_fract++;
       
   498 	if (_date_fract < DAY_TICKS) return;
       
   499 	_date_fract = 0;
       
   500 
       
   501 	/* yeah, increse day counter and call various daily loops */
       
   502 	_date++;
       
   503 
       
   504 	TextMessageDailyLoop();
       
   505 
       
   506 	DisasterDailyLoop();
       
   507 	WaypointsDailyLoop();
       
   508 
       
   509 	if (_game_mode != GM_MENU) {
       
   510 		InvalidateWindowWidget(WC_STATUS_BAR, 0, 0);
       
   511 		EnginesDailyLoop();
       
   512 	}
       
   513 
       
   514 	/* check if we entered a new month? */
       
   515 	ConvertDayToYMD(&ymd, _date);
       
   516 	if ((byte)ymd.month == _cur_month)
       
   517 		return;
       
   518 	_cur_month = ymd.month;
       
   519 
       
   520 	/* yes, call various monthly loops */
       
   521 	if (_game_mode != GM_MENU) {
       
   522 		if (HASBIT(_autosave_months[_opt.autosave], _cur_month)) {
       
   523 			_do_autosave = true;
       
   524 			RedrawAutosave();
       
   525 		}
       
   526 
       
   527 		PlayersMonthlyLoop();
       
   528 		EnginesMonthlyLoop();
       
   529 		TownsMonthlyLoop();
       
   530 		IndustryMonthlyLoop();
       
   531 		StationMonthlyLoop();
       
   532 #ifdef ENABLE_NETWORK
       
   533 		if (_network_server) NetworkServerMonthlyLoop();
       
   534 #endif /* ENABLE_NETWORK */
       
   535 	}
       
   536 
       
   537 	/* check if we entered a new year? */
       
   538 	if ((byte)ymd.year == _cur_year)
       
   539 		return;
       
   540 	_cur_year = ymd.year;
       
   541 
       
   542 	/* yes, call various yearly loops */
       
   543 
       
   544 	PlayersYearlyLoop();
       
   545 	TrainsYearlyLoop();
       
   546 	RoadVehiclesYearlyLoop();
       
   547 	AircraftYearlyLoop();
       
   548 	ShipsYearlyLoop();
       
   549 #ifdef ENABLE_NETWORK
       
   550 	if (_network_server) NetworkServerYearlyLoop();
       
   551 #endif /* ENABLE_NETWORK */
       
   552 
       
   553 	/* check if we reached end of the game (31 dec 2050) */
       
   554 	if (_cur_year == _patches.ending_date - MAX_YEAR_BEGIN_REAL) {
       
   555 			ShowEndGameChart();
       
   556 	/* check if we reached 2090 (MAX_YEAR_END_REAL), that's the maximum year. */
       
   557 	} else if (_cur_year == (MAX_YEAR_END + 1)) {
       
   558 		Vehicle* v;
       
   559 
       
   560 		_cur_year = MAX_YEAR_END;
       
   561 		_date = 62093;
       
   562 		FOR_ALL_VEHICLES(v) {
       
   563 			v->date_of_last_service -= 365; // 1 year is 365 days long
       
   564 		}
       
   565 
       
   566 		/* Because the _date wraps here, and text-messages expire by game-days, we have to clean out
       
   567 		 *  all of them if the date is set back, else those messages will hang for ever */
       
   568 		InitTextMessage();
       
   569 	}
       
   570 
       
   571 	if (_patches.auto_euro) CheckSwitchToEuro();
       
   572 
       
   573 	/* XXX: check if year 2050 was reached */
       
   574 }
       
   575 
   296 
   576 int FindFirstBit(uint32 value)
   297 int FindFirstBit(uint32 value)
   577 {
   298 {
   578 	// This is much faster than the one that was before here.
   299 	// This is much faster than the one that was before here.
   579 	//  Created by Darkvater.. blame him if it is wrong ;)
   300 	//  Created by Darkvater.. blame him if it is wrong ;)