2019-05-27 14:55:01 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2007-07-19 11:47:47 +08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2007 Freescale Semiconductor, Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* Description: QE UCC Gigabit Ethernet Ethtool API Set
|
|
|
|
*
|
|
|
|
* Author: Li Yang <leoli@freescale.com>
|
|
|
|
*
|
2008-05-13 13:41:58 +08:00
|
|
|
* Limitation:
|
2011-05-10 16:16:21 +08:00
|
|
|
* Can only get/set settings of the first queue.
|
2009-04-23 00:21:29 +08:00
|
|
|
* Need to re-open the interface manually after changing some parameters.
|
2007-07-19 11:47:47 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/stddef.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/netdevice.h>
|
|
|
|
#include <linux/etherdevice.h>
|
|
|
|
#include <linux/skbuff.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/dma-mapping.h>
|
|
|
|
#include <linux/ethtool.h>
|
|
|
|
#include <linux/mii.h>
|
|
|
|
#include <linux/phy.h>
|
|
|
|
|
|
|
|
#include <asm/io.h>
|
|
|
|
#include <asm/irq.h>
|
2016-12-25 03:46:01 +08:00
|
|
|
#include <linux/uaccess.h>
|
2007-07-19 11:47:47 +08:00
|
|
|
#include <asm/types.h>
|
|
|
|
|
|
|
|
#include "ucc_geth.h"
|
|
|
|
|
2013-04-14 03:03:19 +08:00
|
|
|
static const char hw_stat_gstrings[][ETH_GSTRING_LEN] = {
|
2007-07-19 11:47:47 +08:00
|
|
|
"tx-64-frames",
|
|
|
|
"tx-65-127-frames",
|
|
|
|
"tx-128-255-frames",
|
|
|
|
"rx-64-frames",
|
|
|
|
"rx-65-127-frames",
|
|
|
|
"rx-128-255-frames",
|
|
|
|
"tx-bytes-ok",
|
|
|
|
"tx-pause-frames",
|
|
|
|
"tx-multicast-frames",
|
|
|
|
"tx-broadcast-frames",
|
|
|
|
"rx-frames",
|
|
|
|
"rx-bytes-ok",
|
|
|
|
"rx-bytes-all",
|
|
|
|
"rx-multicast-frames",
|
|
|
|
"rx-broadcast-frames",
|
|
|
|
"stats-counter-carry",
|
|
|
|
"stats-counter-mask",
|
|
|
|
"rx-dropped-frames",
|
|
|
|
};
|
|
|
|
|
2013-04-14 03:03:19 +08:00
|
|
|
static const char tx_fw_stat_gstrings[][ETH_GSTRING_LEN] = {
|
2007-07-19 11:47:47 +08:00
|
|
|
"tx-single-collision",
|
|
|
|
"tx-multiple-collision",
|
2018-04-28 17:57:07 +08:00
|
|
|
"tx-late-collision",
|
2007-07-19 11:47:47 +08:00
|
|
|
"tx-aborted-frames",
|
|
|
|
"tx-lost-frames",
|
|
|
|
"tx-carrier-sense-errors",
|
|
|
|
"tx-frames-ok",
|
|
|
|
"tx-excessive-differ-frames",
|
|
|
|
"tx-256-511-frames",
|
2008-05-23 18:11:27 +08:00
|
|
|
"tx-512-1023-frames",
|
2007-07-19 11:47:47 +08:00
|
|
|
"tx-1024-1518-frames",
|
|
|
|
"tx-jumbo-frames",
|
|
|
|
};
|
|
|
|
|
2013-04-14 03:03:19 +08:00
|
|
|
static const char rx_fw_stat_gstrings[][ETH_GSTRING_LEN] = {
|
2007-07-19 11:47:47 +08:00
|
|
|
"rx-crc-errors",
|
|
|
|
"rx-alignment-errors",
|
|
|
|
"rx-in-range-length-errors",
|
|
|
|
"rx-out-of-range-length-errors",
|
|
|
|
"rx-too-long-frames",
|
|
|
|
"rx-runt",
|
|
|
|
"rx-very-long-event",
|
|
|
|
"rx-symbol-errors",
|
|
|
|
"rx-busy-drop-frames",
|
|
|
|
"reserved",
|
|
|
|
"reserved",
|
|
|
|
"rx-mismatch-drop-frames",
|
|
|
|
"rx-small-than-64",
|
|
|
|
"rx-256-511-frames",
|
|
|
|
"rx-512-1023-frames",
|
|
|
|
"rx-1024-1518-frames",
|
|
|
|
"rx-jumbo-frames",
|
|
|
|
"rx-mac-error-loss",
|
|
|
|
"rx-pause-frames",
|
|
|
|
"reserved",
|
|
|
|
"rx-vlan-removed",
|
|
|
|
"rx-vlan-replaced",
|
|
|
|
"rx-vlan-inserted",
|
|
|
|
"rx-ip-checksum-errors",
|
|
|
|
};
|
|
|
|
|
|
|
|
#define UEC_HW_STATS_LEN ARRAY_SIZE(hw_stat_gstrings)
|
|
|
|
#define UEC_TX_FW_STATS_LEN ARRAY_SIZE(tx_fw_stat_gstrings)
|
|
|
|
#define UEC_RX_FW_STATS_LEN ARRAY_SIZE(rx_fw_stat_gstrings)
|
|
|
|
|
|
|
|
static int
|
2016-05-01 23:08:09 +08:00
|
|
|
uec_get_ksettings(struct net_device *netdev, struct ethtool_link_ksettings *cmd)
|
2007-07-19 11:47:47 +08:00
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
struct phy_device *phydev = ugeth->phydev;
|
|
|
|
|
|
|
|
if (!phydev)
|
|
|
|
return -ENODEV;
|
|
|
|
|
2017-06-13 15:09:46 +08:00
|
|
|
phy_ethtool_ksettings_get(phydev, cmd);
|
|
|
|
|
|
|
|
return 0;
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2016-05-01 23:08:09 +08:00
|
|
|
uec_set_ksettings(struct net_device *netdev,
|
|
|
|
const struct ethtool_link_ksettings *cmd)
|
2007-07-19 11:47:47 +08:00
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
struct phy_device *phydev = ugeth->phydev;
|
|
|
|
|
|
|
|
if (!phydev)
|
|
|
|
return -ENODEV;
|
|
|
|
|
2016-05-01 23:08:09 +08:00
|
|
|
return phy_ethtool_ksettings_set(phydev, cmd);
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uec_get_pauseparam(struct net_device *netdev,
|
|
|
|
struct ethtool_pauseparam *pause)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
|
|
|
|
pause->autoneg = ugeth->phydev->autoneg;
|
|
|
|
|
|
|
|
if (ugeth->ug_info->receiveFlowControl)
|
|
|
|
pause->rx_pause = 1;
|
|
|
|
if (ugeth->ug_info->transmitFlowControl)
|
|
|
|
pause->tx_pause = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
uec_set_pauseparam(struct net_device *netdev,
|
|
|
|
struct ethtool_pauseparam *pause)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
ugeth->ug_info->receiveFlowControl = pause->rx_pause;
|
|
|
|
ugeth->ug_info->transmitFlowControl = pause->tx_pause;
|
2008-05-13 13:41:58 +08:00
|
|
|
|
2007-07-19 11:47:47 +08:00
|
|
|
if (ugeth->phydev->autoneg) {
|
|
|
|
if (netif_running(netdev)) {
|
|
|
|
/* FIXME: automatically restart */
|
2013-04-14 03:03:19 +08:00
|
|
|
netdev_info(netdev, "Please re-open the interface\n");
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
struct ucc_geth_info *ug_info = ugeth->ug_info;
|
|
|
|
|
|
|
|
ret = init_flow_control_params(ug_info->aufc,
|
|
|
|
ug_info->receiveFlowControl,
|
|
|
|
ug_info->transmitFlowControl,
|
|
|
|
ug_info->pausePeriod,
|
|
|
|
ug_info->extensionField,
|
|
|
|
&ugeth->uccf->uf_regs->upsmr,
|
|
|
|
&ugeth->ug_regs->uempr,
|
|
|
|
&ugeth->ug_regs->maccfg1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
uec_get_msglevel(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
return ugeth->msg_enable;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uec_set_msglevel(struct net_device *netdev, uint32_t data)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
ugeth->msg_enable = data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
uec_get_regs_len(struct net_device *netdev)
|
|
|
|
{
|
|
|
|
return sizeof(struct ucc_geth);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uec_get_regs(struct net_device *netdev,
|
|
|
|
struct ethtool_regs *regs, void *p)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
u32 __iomem *ug_regs = (u32 __iomem *)ugeth->ug_regs;
|
|
|
|
u32 *buff = p;
|
|
|
|
|
|
|
|
for (i = 0; i < sizeof(struct ucc_geth) / sizeof(u32); i++)
|
|
|
|
buff[i] = in_be32(&ug_regs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uec_get_ringparam(struct net_device *netdev,
|
|
|
|
struct ethtool_ringparam *ring)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
struct ucc_geth_info *ug_info = ugeth->ug_info;
|
|
|
|
int queue = 0;
|
|
|
|
|
|
|
|
ring->rx_max_pending = UCC_GETH_BD_RING_SIZE_MAX;
|
|
|
|
ring->rx_mini_max_pending = UCC_GETH_BD_RING_SIZE_MAX;
|
|
|
|
ring->rx_jumbo_max_pending = UCC_GETH_BD_RING_SIZE_MAX;
|
|
|
|
ring->tx_max_pending = UCC_GETH_BD_RING_SIZE_MAX;
|
|
|
|
|
|
|
|
ring->rx_pending = ug_info->bdRingLenRx[queue];
|
|
|
|
ring->rx_mini_pending = ug_info->bdRingLenRx[queue];
|
|
|
|
ring->rx_jumbo_pending = ug_info->bdRingLenRx[queue];
|
|
|
|
ring->tx_pending = ug_info->bdRingLenTx[queue];
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
uec_set_ringparam(struct net_device *netdev,
|
|
|
|
struct ethtool_ringparam *ring)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
struct ucc_geth_info *ug_info = ugeth->ug_info;
|
|
|
|
int queue = 0, ret = 0;
|
|
|
|
|
|
|
|
if (ring->rx_pending < UCC_GETH_RX_BD_RING_SIZE_MIN) {
|
2013-04-14 03:03:19 +08:00
|
|
|
netdev_info(netdev, "RxBD ring size must be no smaller than %d\n",
|
|
|
|
UCC_GETH_RX_BD_RING_SIZE_MIN);
|
2007-07-19 11:47:47 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (ring->rx_pending % UCC_GETH_RX_BD_RING_SIZE_ALIGNMENT) {
|
2013-04-14 03:03:19 +08:00
|
|
|
netdev_info(netdev, "RxBD ring size must be multiple of %d\n",
|
|
|
|
UCC_GETH_RX_BD_RING_SIZE_ALIGNMENT);
|
2007-07-19 11:47:47 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (ring->tx_pending < UCC_GETH_TX_BD_RING_SIZE_MIN) {
|
2013-04-14 03:03:19 +08:00
|
|
|
netdev_info(netdev, "TxBD ring size must be no smaller than %d\n",
|
|
|
|
UCC_GETH_TX_BD_RING_SIZE_MIN);
|
2007-07-19 11:47:47 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-05-03 21:33:23 +08:00
|
|
|
if (netif_running(netdev))
|
|
|
|
return -EBUSY;
|
|
|
|
|
2007-07-19 11:47:47 +08:00
|
|
|
ug_info->bdRingLenRx[queue] = ring->rx_pending;
|
|
|
|
ug_info->bdRingLenTx[queue] = ring->tx_pending;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-10-04 09:07:32 +08:00
|
|
|
static int uec_get_sset_count(struct net_device *netdev, int sset)
|
2007-07-19 11:47:47 +08:00
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
u32 stats_mode = ugeth->ug_info->statisticsMode;
|
|
|
|
int len = 0;
|
|
|
|
|
2007-10-04 09:07:32 +08:00
|
|
|
switch (sset) {
|
|
|
|
case ETH_SS_STATS:
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE)
|
|
|
|
len += UEC_HW_STATS_LEN;
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX)
|
|
|
|
len += UEC_TX_FW_STATS_LEN;
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX)
|
|
|
|
len += UEC_RX_FW_STATS_LEN;
|
|
|
|
|
|
|
|
return len;
|
2007-07-19 11:47:47 +08:00
|
|
|
|
2007-10-04 09:07:32 +08:00
|
|
|
default:
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void uec_get_strings(struct net_device *netdev, u32 stringset, u8 *buf)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
u32 stats_mode = ugeth->ug_info->statisticsMode;
|
|
|
|
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE) {
|
|
|
|
memcpy(buf, hw_stat_gstrings, UEC_HW_STATS_LEN *
|
|
|
|
ETH_GSTRING_LEN);
|
|
|
|
buf += UEC_HW_STATS_LEN * ETH_GSTRING_LEN;
|
|
|
|
}
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX) {
|
|
|
|
memcpy(buf, tx_fw_stat_gstrings, UEC_TX_FW_STATS_LEN *
|
|
|
|
ETH_GSTRING_LEN);
|
|
|
|
buf += UEC_TX_FW_STATS_LEN * ETH_GSTRING_LEN;
|
|
|
|
}
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX)
|
2008-05-23 18:11:26 +08:00
|
|
|
memcpy(buf, rx_fw_stat_gstrings, UEC_RX_FW_STATS_LEN *
|
2007-07-19 11:47:47 +08:00
|
|
|
ETH_GSTRING_LEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void uec_get_ethtool_stats(struct net_device *netdev,
|
|
|
|
struct ethtool_stats *stats, uint64_t *data)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
u32 stats_mode = ugeth->ug_info->statisticsMode;
|
|
|
|
u32 __iomem *base;
|
|
|
|
int i, j = 0;
|
|
|
|
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE) {
|
2009-08-27 15:35:47 +08:00
|
|
|
if (ugeth->ug_regs)
|
|
|
|
base = (u32 __iomem *)&ugeth->ug_regs->tx64;
|
|
|
|
else
|
|
|
|
base = NULL;
|
|
|
|
|
2007-07-19 11:47:47 +08:00
|
|
|
for (i = 0; i < UEC_HW_STATS_LEN; i++)
|
2009-08-27 15:35:47 +08:00
|
|
|
data[j++] = base ? in_be32(&base[i]) : 0;
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX) {
|
|
|
|
base = (u32 __iomem *)ugeth->p_tx_fw_statistics_pram;
|
|
|
|
for (i = 0; i < UEC_TX_FW_STATS_LEN; i++)
|
2008-11-14 02:26:27 +08:00
|
|
|
data[j++] = base ? in_be32(&base[i]) : 0;
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX) {
|
|
|
|
base = (u32 __iomem *)ugeth->p_rx_fw_statistics_pram;
|
|
|
|
for (i = 0; i < UEC_RX_FW_STATS_LEN; i++)
|
2008-11-14 02:26:27 +08:00
|
|
|
data[j++] = base ? in_be32(&base[i]) : 0;
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Report driver information */
|
|
|
|
static void
|
|
|
|
uec_get_drvinfo(struct net_device *netdev,
|
|
|
|
struct ethtool_drvinfo *drvinfo)
|
|
|
|
{
|
2013-01-06 08:44:26 +08:00
|
|
|
strlcpy(drvinfo->driver, DRV_NAME, sizeof(drvinfo->driver));
|
|
|
|
strlcpy(drvinfo->bus_info, "QUICC ENGINE", sizeof(drvinfo->bus_info));
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|
|
|
|
|
2009-08-27 15:35:57 +08:00
|
|
|
#ifdef CONFIG_PM
|
|
|
|
|
|
|
|
static void uec_get_wol(struct net_device *netdev, struct ethtool_wolinfo *wol)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
struct phy_device *phydev = ugeth->phydev;
|
|
|
|
|
|
|
|
if (phydev && phydev->irq)
|
|
|
|
wol->supported |= WAKE_PHY;
|
|
|
|
if (qe_alive_during_sleep())
|
|
|
|
wol->supported |= WAKE_MAGIC;
|
|
|
|
|
|
|
|
wol->wolopts = ugeth->wol_en;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int uec_set_wol(struct net_device *netdev, struct ethtool_wolinfo *wol)
|
|
|
|
{
|
|
|
|
struct ucc_geth_private *ugeth = netdev_priv(netdev);
|
|
|
|
struct phy_device *phydev = ugeth->phydev;
|
|
|
|
|
|
|
|
if (wol->wolopts & ~(WAKE_PHY | WAKE_MAGIC))
|
|
|
|
return -EINVAL;
|
|
|
|
else if (wol->wolopts & WAKE_PHY && (!phydev || !phydev->irq))
|
|
|
|
return -EINVAL;
|
|
|
|
else if (wol->wolopts & WAKE_MAGIC && !qe_alive_during_sleep())
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
ugeth->wol_en = wol->wolopts;
|
|
|
|
device_set_wakeup_enable(&netdev->dev, ugeth->wol_en);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
#define uec_get_wol NULL
|
|
|
|
#define uec_set_wol NULL
|
|
|
|
#endif /* CONFIG_PM */
|
|
|
|
|
2007-07-19 11:47:47 +08:00
|
|
|
static const struct ethtool_ops uec_ethtool_ops = {
|
|
|
|
.get_drvinfo = uec_get_drvinfo,
|
|
|
|
.get_regs_len = uec_get_regs_len,
|
|
|
|
.get_regs = uec_get_regs,
|
|
|
|
.get_msglevel = uec_get_msglevel,
|
|
|
|
.set_msglevel = uec_set_msglevel,
|
2016-11-16 02:06:41 +08:00
|
|
|
.nway_reset = phy_ethtool_nway_reset,
|
2007-07-19 11:47:47 +08:00
|
|
|
.get_link = ethtool_op_get_link,
|
|
|
|
.get_ringparam = uec_get_ringparam,
|
|
|
|
.set_ringparam = uec_set_ringparam,
|
|
|
|
.get_pauseparam = uec_get_pauseparam,
|
|
|
|
.set_pauseparam = uec_set_pauseparam,
|
2007-10-04 09:07:32 +08:00
|
|
|
.get_sset_count = uec_get_sset_count,
|
2007-07-19 11:47:47 +08:00
|
|
|
.get_strings = uec_get_strings,
|
|
|
|
.get_ethtool_stats = uec_get_ethtool_stats,
|
2009-08-27 15:35:57 +08:00
|
|
|
.get_wol = uec_get_wol,
|
|
|
|
.set_wol = uec_set_wol,
|
2012-04-04 06:59:40 +08:00
|
|
|
.get_ts_info = ethtool_op_get_ts_info,
|
2016-05-01 23:08:09 +08:00
|
|
|
.get_link_ksettings = uec_get_ksettings,
|
|
|
|
.set_link_ksettings = uec_set_ksettings,
|
2007-07-19 11:47:47 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
void uec_set_ethtool_ops(struct net_device *netdev)
|
|
|
|
{
|
2014-05-11 08:12:32 +08:00
|
|
|
netdev->ethtool_ops = &uec_ethtool_ops;
|
2007-07-19 11:47:47 +08:00
|
|
|
}
|