792a5d290a
Signed-off-by: Alexey I. Froloff <raorn@raorn.name>
602 lines
18 KiB
C
602 lines
18 KiB
C
/***************************************************************************
|
||
wmpower.c - description
|
||
-------------------
|
||
begin : Feb 10 2003
|
||
copyright : (C) 2003,2004,2005 by Noberasco Michele
|
||
e-mail : noberasco.gnu@disi.unige.it
|
||
***************************************************************************/
|
||
|
||
/***************************************************************************
|
||
* *
|
||
* 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., *
|
||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. *
|
||
* *
|
||
***************************************************************************/
|
||
|
||
/***************************************************************************
|
||
Many thanks to Filippo Panessa for his wmab...
|
||
it's code was the base for this program
|
||
***************************************************************************/
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include <unistd.h>
|
||
#include <string.h>
|
||
#include <time.h>
|
||
#include <X11/X.h>
|
||
#include <X11/xpm.h>
|
||
|
||
#include "open_syslog_on_stderr.h"
|
||
#include "power_management.h"
|
||
#include "dockapp.h"
|
||
#include "wmpower_master.xpm"
|
||
#include "wmpower_master_LowColor.xpm"
|
||
#include "wmpower_mask.xbm"
|
||
|
||
void ParseCMDLine (int argc, char *argv[]);
|
||
void ShowACstatus(int ac_on_line);
|
||
void ShowFanStatus(int fanstatus);
|
||
void ShowTemperature(int temp, int is_celsius);
|
||
void ShowChargeStatus(int charging);
|
||
void ShowBatteryTime(int time, int percentage, int charging, int ac_on_line);
|
||
void ShowBatteryPercentage(int percentage);
|
||
void ShowBatteryLed(int present, int percentage, int ac_on_line);
|
||
|
||
int no_meddling = 0; /* Should we stop managing power status? */
|
||
int no_full_battery = 0; /* Should we always use max power when plugged? */
|
||
|
||
int CriticalLevel = 10; /* Battery critical level */
|
||
int LowLevel = 40; /* Battery low level */
|
||
|
||
#define CMDLINELEN 512
|
||
int WarnTime = 2; /* When to execute the warn command */
|
||
char WarnCommand[CMDLINELEN] = ""; /* The warn command to execute */
|
||
|
||
float BlinkRate = 3.00; /* blinks per second */
|
||
|
||
/* Controls beeping when you get to critical */
|
||
/* battery level: Off by default */
|
||
int Beep = 0; /* to beep or not to beep? */
|
||
int Volume = 50; /* ring bell at 50% volume */
|
||
|
||
/* Mouse wheel */
|
||
unsigned int wheel_button_up = 4;
|
||
unsigned int wheel_button_down = 5;
|
||
|
||
/* Monitor first battery by default */
|
||
int our_battery = 1;
|
||
|
||
/* Use a lower number of colors for the poor saps on */
|
||
/* 8-bit displays -- common on laptops! */
|
||
int UseLowColorPixmap = 0;
|
||
|
||
int main (int argc, char *argv[])
|
||
{
|
||
pm_status power_status;
|
||
XEvent event;
|
||
int fbc_toggle=1, fbc_auto=1;
|
||
int old_battery_charging;
|
||
time_t polling = 0;
|
||
struct timespec delay; /* pause between interface updates */
|
||
char Command[CMDLINELEN+3];
|
||
int warned = 0;
|
||
|
||
delay.tv_sec = 0;
|
||
delay.tv_nsec = 500000000;
|
||
|
||
BlinkRate = (BlinkRate >= 0.0) ? BlinkRate : -1.0 * BlinkRate;
|
||
waittime = 0; /* /proc polling interval */
|
||
minBrightness = -1;
|
||
maxBrightness = -1;
|
||
|
||
fprintf(stderr, "\nWelcome to wmpower version %s...\n", VERSION);
|
||
|
||
cpufreq_online_governor = NULL;
|
||
cpufreq_offline_governor = NULL;
|
||
|
||
/* Parse any command line arguments. */
|
||
ParseCMDLine (argc, argv);
|
||
|
||
/* Check for Power Management support */
|
||
if (!pm_support(our_battery))
|
||
{
|
||
fprintf (stderr, "\nNo power management support...\n");
|
||
return EXIT_FAILURE;
|
||
}
|
||
|
||
/* Create window of the program */
|
||
if (UseLowColorPixmap) openXwindow (argc, argv, wmpower_master_LowColor, (char *) wmpower_mask_bits, wmpower_mask_width, wmpower_mask_height);
|
||
else openXwindow (argc, argv, wmpower_master, (char *) wmpower_mask_bits, wmpower_mask_width, wmpower_mask_height);
|
||
|
||
/* Loop until we die... */
|
||
while (1)
|
||
{
|
||
/* Get current power status */
|
||
old_battery_charging = power_status.battery_charging;
|
||
if (!waittime) get_power_status(&power_status);
|
||
else if ((time(NULL)-polling) >= waittime)
|
||
{
|
||
get_power_status(&power_status);
|
||
polling = time(NULL);
|
||
}
|
||
|
||
/* Manage power features only if function is not disabled */
|
||
if (!no_meddling)
|
||
{
|
||
/* Re-enable auto power mode switching whan battery status changes */
|
||
if (old_battery_charging != power_status.battery_charging) fbc_auto = 1;
|
||
|
||
/* Enable fast battery charge mode if on AC and batt is charging */
|
||
if (!no_full_battery && power_status.ac_on_line && power_status.battery_charging && fbc_auto && !fbc_toggle && !(power_status.battery_percentage == 100))
|
||
{
|
||
fast_battery_charge(1);
|
||
fbc_toggle = 1;
|
||
fbc_auto = 1;
|
||
}
|
||
|
||
/* Adjust variables value when battery reaches 100% */
|
||
if (fbc_toggle && (power_status.battery_percentage == 100))
|
||
{
|
||
fast_battery_charge(0);
|
||
fbc_toggle = 0;
|
||
fbc_auto = 1;
|
||
}
|
||
|
||
/* If battery not present and fast charge mode, disable it */
|
||
if (fbc_toggle && !(power_status.battery_present))
|
||
{
|
||
fast_battery_charge(0);
|
||
fbc_toggle = 0;
|
||
fbc_auto = 1;
|
||
}
|
||
|
||
/* Set various pm features whenever applicable */
|
||
set_pm_features();
|
||
}
|
||
|
||
/* Execute the warning command, if needed */
|
||
if (WarnCommand && *WarnCommand && !power_status.ac_on_line && !warned
|
||
&& power_status.battery_time <= WarnTime)
|
||
{
|
||
warned = 1;
|
||
sprintf(Command, "%s &", WarnCommand);
|
||
system(Command);
|
||
}
|
||
if (power_status.ac_on_line)
|
||
warned = 0;
|
||
|
||
/* Show AC status led */
|
||
ShowACstatus(power_status.ac_on_line);
|
||
|
||
/* Display FAN status. */
|
||
ShowFanStatus(power_status.fan_status);
|
||
|
||
/* Display temperature. */
|
||
ShowTemperature(power_status.temperature, power_status.temp_is_celsius);
|
||
|
||
/* Display charge status */
|
||
ShowChargeStatus(power_status.battery_charging);
|
||
|
||
/* Display the "Time Left" */
|
||
ShowBatteryTime(power_status.battery_time, power_status.battery_percentage, power_status.battery_charging, power_status.ac_on_line);
|
||
|
||
/* Display battery percentage */
|
||
ShowBatteryPercentage(power_status.battery_percentage);
|
||
|
||
/* Display battery status led */
|
||
ShowBatteryLed(power_status.battery_present, power_status.battery_percentage, power_status.ac_on_line);
|
||
|
||
/* Process any pending X events. */
|
||
while (XPending (display))
|
||
{
|
||
XNextEvent (display, &event);
|
||
switch (event.type)
|
||
{
|
||
case Expose:
|
||
RedrawWindow ();
|
||
continue;
|
||
case ButtonPress:
|
||
if (no_meddling)
|
||
{
|
||
fprintf(stderr, "You cannot change PM status in '-no-meddling' mode of operation\n");
|
||
continue;
|
||
}
|
||
if (event.xbutton.button == wheel_button_up)
|
||
{
|
||
lcdBrightness_UpOneStep();
|
||
continue;
|
||
}
|
||
if (event.xbutton.button == wheel_button_down)
|
||
{
|
||
lcdBrightness_DownOneStep();
|
||
continue;
|
||
}
|
||
fbc_toggle = !get_fast_battery_charge_mode();
|
||
fbc_auto = 0;
|
||
fast_battery_charge(fbc_toggle);
|
||
continue;
|
||
case ButtonRelease:
|
||
continue;
|
||
}
|
||
}
|
||
|
||
/* Redraw and wait for next update */
|
||
RedrawWindow ();
|
||
nanosleep(&delay, NULL);
|
||
}
|
||
}
|
||
|
||
|
||
/* Show AC status led */
|
||
void ShowACstatus(int ac_on_line)
|
||
{
|
||
/* Check AC status. */
|
||
if (ac_on_line)
|
||
/* AC on-line. I.e. we are "plugged-in". */
|
||
copyXPMArea (68, 6, 12, 7, 31, 35);
|
||
else
|
||
/* AC off-line. I.e. we are using battery. */
|
||
copyXPMArea (68, 20, 12, 7, 31, 35);
|
||
}
|
||
|
||
|
||
|
||
/* Display fan status */
|
||
void ShowFanStatus(int fan_status)
|
||
{
|
||
if (fan_status == PM_Error)
|
||
{
|
||
/* Plot the red - Symbol */
|
||
copyXPMArea (165, 60, 6, 7, 23, 50);
|
||
return;
|
||
}
|
||
|
||
/* Plot fan status: 0 not active, 1 running */
|
||
copyXPMArea (fan_status * 6 + 4, 69, 6, 7, 23, 50);
|
||
}
|
||
|
||
|
||
|
||
/* Display charge status */
|
||
void ShowChargeStatus(int charging)
|
||
{
|
||
/* Paste up the default charge status and time */
|
||
copyXPMArea ( 83, 93, 41, 9, 15, 7);
|
||
copyXPMArea (104, 6, 5, 7, 6, 7);
|
||
|
||
/* Check to see if we are charging. */
|
||
if (charging)
|
||
/* Battery Status: Charging. */
|
||
copyXPMArea (82, 68, 7, 9, 6, 7);
|
||
else
|
||
/* Battery Status: NOT Charging. */
|
||
copyXPMArea (88, 68, 7, 9, 6, 7);
|
||
}
|
||
|
||
|
||
|
||
/* Display battery status led */
|
||
void ShowBatteryLed(int present, int percentage, int ac_on_line)
|
||
{
|
||
static int Toggle; /* Switch for battery led blinking */
|
||
|
||
if (!present)
|
||
{
|
||
copyXPMArea (95, 19, 16, 10, 43, 34);
|
||
return;
|
||
}
|
||
|
||
/* Battery Status: Critical. */
|
||
/* Blink the red led on/off... */
|
||
if (percentage <= CriticalLevel && !ac_on_line)
|
||
{
|
||
if (Toggle || (BlinkRate == 0.0))
|
||
{
|
||
if (Beep && !ac_on_line) XBell (display, Volume);
|
||
Toggle = 0;
|
||
copyXPMArea (4, 105, 16, 10, 43, 34);
|
||
}
|
||
else
|
||
{
|
||
Toggle = 1;
|
||
copyXPMArea (58, 105, 16, 10, 43, 34);
|
||
}
|
||
return;
|
||
}
|
||
|
||
/* Battery Status: Low. */
|
||
/* Fixed yellow led */
|
||
if (percentage <= LowLevel)
|
||
{
|
||
copyXPMArea (22, 105, 16, 10, 43, 34);
|
||
return;
|
||
}
|
||
|
||
/* Battery Status: Normal. */
|
||
/* Fixed blue led */
|
||
copyXPMArea (40, 105, 16, 10, 43, 34);
|
||
}
|
||
|
||
|
||
|
||
/* Display Temperature */
|
||
void ShowTemperature(int temp, int temp_is_celsius)
|
||
{
|
||
/* PM_Error getting temperature value */
|
||
/* or value out of range */
|
||
if ( (temp < 0) || (temp > 99) )
|
||
{
|
||
/* Plot PM_Error message */
|
||
copyXPMArea (165, 60, 6, 7, 33, 50);
|
||
copyXPMArea (165, 60, 6, 7, 39, 50);
|
||
copyXPMArea (135, 60, 6, 7, 45, 50);
|
||
copyXPMArea ( 68, 69, 6, 7, 51, 50);
|
||
return;
|
||
}
|
||
|
||
/* Plot temperature */
|
||
if (temp < 10) copyXPMArea ((temp) * 6 + 4, 69, 6, 7, 39, 50);
|
||
else
|
||
{
|
||
copyXPMArea ((temp / 10) * 6 + 4, 69, 6, 7, 33, 50);
|
||
copyXPMArea ((temp % 10) * 6 + 4, 69, 6, 7, 39, 50);
|
||
}
|
||
|
||
/* Plot the <20> Symbol */
|
||
copyXPMArea (135, 60, 6, 7, 45, 50);
|
||
|
||
/* Plot the C Symbol */
|
||
if (temp_is_celsius) copyXPMArea (68, 69, 6, 7, 51, 50);
|
||
}
|
||
|
||
|
||
|
||
/* Display the "Time Left". This time means: */
|
||
/* If not charging: Time left before battery drains to 0% */
|
||
/* If charging: Time left before battery gets to maximum */
|
||
void ShowBatteryTime(int time, int percentage, int charging, int ac_on_line)
|
||
{
|
||
int battery_time=time;
|
||
int hour, min;
|
||
|
||
if ( (battery_time < -1) || ((battery_time == 0)&&(percentage == 0)) || (ac_on_line&&(percentage == 100)) )
|
||
{
|
||
/* In case battery is fully charged and we are on AC power,
|
||
* or there is some problem reading battery time
|
||
* we display a "null" indicator (--:--)
|
||
*/
|
||
copyXPMArea (83, 106, 41, 9, 15, 7);
|
||
return;
|
||
}
|
||
|
||
/* Now we are sure battery time is consistent */
|
||
if (percentage == 100) battery_time = 0;
|
||
hour = battery_time / 60;
|
||
min = battery_time % 60;
|
||
|
||
/* show '-' sign when charging, '+' otherwise */
|
||
if (charging)
|
||
copyXPMArea (83, 106, 41, 9, 15, 7);
|
||
else
|
||
copyXPMArea (83, 93, 41, 9, 15, 7);
|
||
|
||
/* Show 10's (hour) */
|
||
copyXPMArea ((hour / 10) * 7 + 5, 93, 7, 9, 21, 7);
|
||
|
||
/* Show 1's (hour) */
|
||
copyXPMArea ((hour % 10) * 7 + 5, 93, 7, 9, 29, 7);
|
||
|
||
/* colon */
|
||
copyXPMArea (76, 93, 2, 9, 38, 7);
|
||
|
||
/* Show 10's (min) */
|
||
copyXPMArea ((min / 10) * 7 + 5, 93, 7, 9, 42, 7);
|
||
|
||
/* Show 1's (min) */
|
||
copyXPMArea ((min % 10) * 7 + 5, 93, 7, 9, 50, 7);
|
||
}
|
||
|
||
|
||
|
||
/* Display battery percentage */
|
||
void ShowBatteryPercentage(int percentage)
|
||
{
|
||
int k;
|
||
|
||
copyXPMArea (76, 81, 19, 7, 7, 34); /* Show Default % */
|
||
copyXPMArea (66, 31, 49, 9, 7, 21); /* Show Default Meter */
|
||
|
||
if (percentage == 100)
|
||
{
|
||
/* If 100%, show 100% */
|
||
copyXPMArea (15, 81, 1, 7, 7, 34);
|
||
copyXPMArea ( 5, 81, 6, 7, 9, 34);
|
||
copyXPMArea ( 5, 81, 6, 7, 15, 34);
|
||
copyXPMArea (64, 81, 7, 7, 21, 34); /* Show '%' */
|
||
|
||
/* Show rainbow battery bar */
|
||
copyXPMArea (66, 52, 49, 9, 7, 21);
|
||
|
||
return;
|
||
}
|
||
/* Show 10's */
|
||
if (percentage >= 10) copyXPMArea ((percentage / 10) * 6 + 4, 81, 6, 7, 9, 34);
|
||
|
||
/* Show 1's */
|
||
copyXPMArea ((percentage % 10) * 6 + 4, 81, 6, 7, 15, 34);
|
||
|
||
/* Show '%' */
|
||
copyXPMArea (64, 81, 7, 7, 21, 34);
|
||
|
||
/* Show Meter */
|
||
k = percentage * 49 / 100;
|
||
|
||
/* Show rainbow battery bar */
|
||
copyXPMArea (66, 52, k, 9, 7, 21);
|
||
if (k % 2) copyXPMArea (66 + k - 1, 52, 1, 9, 7 + k - 1, 21);
|
||
else copyXPMArea (66 + k, 52, 1, 9, 7 + k, 21);
|
||
}
|
||
|
||
|
||
|
||
/* Show message about usage */
|
||
void message(void)
|
||
{
|
||
printf("\nwmpower is a tool for checking and setting power management status for");
|
||
printf("\nlaptop computers. Right now is supports both APM and APCI enabled");
|
||
printf("\nkernels, plus special support for Toshiba and Compal hardware.");
|
||
printf("\n\nUsage: wmpower [options]\n");
|
||
printf("\n\nOptions:\n");
|
||
printf("\t-no-meddling\t\tDon't manage power status, just show info.\n");
|
||
printf("\t-no-full-battery\tDon't wait for 100%% battery before going back\n");
|
||
printf("\t\t\t\tto full power.\n");
|
||
printf("\t-no-cpufreq\t\tDon't scale CPU frequency according to power status.\n");
|
||
printf("\t-no-noflushd\t\tDisable use of \"noflushd\" daemon:\n");
|
||
printf("\t\t\t\tnoflushd is a tool for managing spin-down\n");
|
||
printf("\t\t\t\tof hard disks after a certain amount of time\n");
|
||
printf("\t\t\t\tsee <http://noflushd.sourceforge.net> for details.\n");
|
||
printf("\t-no-toshiba\t\tDisable direct access to toshiba hardware,\n");
|
||
printf("\t\t\t\tuse only generic ACPI/APM calls instead.\n");
|
||
printf("\t\t\t\tThis is recommended on newer toshibas.\n");
|
||
printf("\t-battery <num>\t\tMonitor your nth battery instead of first one.\n");
|
||
printf("\t-display <display>\tUse alternate display.\n");
|
||
printf("\t-geometry <geometry>\twmpower window geometry.\n");
|
||
printf("\t-l\t\t\tUse a low-color pixmap.\n");
|
||
printf("\t-L <LowLevel>\t\tDefine level at which yellow LED turns on.\n");
|
||
printf("\t-C <CriticalLevel>\tDefine level at which red LED turns on.\n");
|
||
printf("\t-B <Volume>\t\tBeep at Critical Level (-100%% to 100%%).\n");
|
||
printf("\t-w <command>\t\tWarn command to run when remaining time is low.\n");
|
||
printf("\t-W <minutes>\t\tMinutes of remaining time when to run warn command.\n");
|
||
printf("\t-u <seconds>\t\tSet wmpower polling interval.\n");
|
||
printf("\t-m <brightness>\t\tUse this LCD brightness value while running on battery power.\n");
|
||
printf("\t-M <brightness>\t\tUse this LCD brightness value while running on AC power.\n");
|
||
printf("\t-g <governor>\t\tUse this CPUFreq scaling governor while running on battery power.\n");
|
||
printf("\t-G <governor>\t\tUse this CPUFreq scaling governor while running on AC power.\n");
|
||
printf("\t-s\t\t\tMake wmpower log to syslog instead of standard error.\n");
|
||
printf("\t-h\t\t\tDisplay this help screen.\n");
|
||
printf("\nClicking on program window at run-time overrides any option,");
|
||
printf("\nthus switching between low-power and full-power modes.");
|
||
printf("\nYou can use the mouse wheel to adjust your lcd brightness.\n\n");
|
||
|
||
exit(EXIT_FAILURE);
|
||
}
|
||
|
||
|
||
/* Parse command line arguments */
|
||
void ParseCMDLine (int argc, char *argv[])
|
||
{
|
||
char *cmdline;
|
||
int i;
|
||
|
||
for (i = 1; i < argc; i++)
|
||
{
|
||
cmdline = argv[i];
|
||
if (cmdline[0] == '-')
|
||
{
|
||
switch (cmdline[1])
|
||
{
|
||
case 'b':
|
||
if (!strcmp(cmdline, "-battery"))
|
||
{
|
||
if (argc == i+1) message();
|
||
our_battery = atoi(argv[++i]);
|
||
if (our_battery < 1) message();
|
||
}
|
||
else message();
|
||
break;
|
||
case 'd':
|
||
++i;
|
||
break;
|
||
case 'C':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
CriticalLevel = atoi (argv[++i]);
|
||
break;
|
||
case 'g':
|
||
if ( !strcmp(cmdline, "-geometry"))
|
||
{
|
||
extern char *Geometry;
|
||
if ( argc == i+1 ) message();
|
||
Geometry = argv[++i];
|
||
break;
|
||
}
|
||
if (cmdline[2] != '\0') message();
|
||
cpufreq_offline_governor = argv[++i];
|
||
break;
|
||
case 'G':
|
||
if (cmdline[2] != '\0') message();
|
||
cpufreq_online_governor = argv[++i];
|
||
break;
|
||
case 'L':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
LowLevel = atoi (argv[++i]);
|
||
break;
|
||
case 'l':
|
||
if (cmdline[2] != '\0') message();
|
||
UseLowColorPixmap = 1;
|
||
break;
|
||
case 'u':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
waittime = atoi (argv[++i]);
|
||
if (waittime <= 0) message();
|
||
fprintf(stderr, "Polling time: %d second%s.\n", waittime, (waittime == 1)? "" : "s");
|
||
break;
|
||
case 'B':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
Beep = 1;
|
||
Volume = atoi (argv[++i]);
|
||
break;
|
||
case 'm':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
minBrightness = atoi (argv[++i]);
|
||
break;
|
||
case 'M':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
maxBrightness = atoi (argv[++i]);
|
||
break;
|
||
case 'w':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
strncpy(WarnCommand, argv[++i], CMDLINELEN-1);
|
||
break;
|
||
case 'W':
|
||
if (cmdline[2] != '\0') message();
|
||
if (argc == i+1) message();
|
||
WarnTime = atoi (argv[++i]);
|
||
break;
|
||
case 's':
|
||
if (cmdline[2] != '\0') message();
|
||
fprintf(stderr, "Switching to syslog logging...\n");
|
||
open_syslog_on_stderr();
|
||
break;
|
||
case 'n':
|
||
if (!strcmp(cmdline, "-no-meddling")) {no_meddling = 1; break;}
|
||
if (!strcmp(cmdline, "-no-full-battery")) {no_full_battery = 1; break;}
|
||
if (!strcmp(cmdline, "-no-noflushd")) {set_noflushd_use(0); break;}
|
||
if (!strcmp(cmdline, "-no-toshiba")) {set_toshiba_hardware_use(0); break;}
|
||
if (!strcmp(cmdline, "-no-cpufreq")) {set_cpufreq_use(0); break;}
|
||
default:
|
||
message();
|
||
}
|
||
}
|
||
else message();
|
||
}
|
||
}
|