1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 <2025> | Index | 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 <2025> |
<== Date ==> | <== Thread ==> |
---|
Subject: | Re: Understanding ASYN_CANBLOCK |
From: | Marco Filho via Tech-talk <tech-talk at aps.anl.gov> |
To: | Mark Rivers <rivers at cars.uchicago.edu>, "tech-talk at aps.anl.gov" <tech-talk at aps.anl.gov> |
Date: | Fri, 31 Jan 2025 11:11:19 +0000 |
Hi, Mark and thanks for the help RMM::RMM(const char *portName, const char *addressIP, int addressPort, const char *configFile)
: asynPortDriver(portName, 1, asynInt8ArrayMask | asynInt32Mask | asynInt32ArrayMask | asynInt64Mask | asynFloat64Mask | asynOctetMask | asynDrvUserMask, // Interfaces that we implement asynInt8ArrayMask | asynInt32Mask | asynInt32ArrayMask | asynInt64Mask | asynFloat64Mask | asynOctetMask, // Interfaces that do callbacks ASYN_MULTIDEVICE | ASYN_CANBLOCK, 1, /* ASYN_CANBLOCK=1, ASYN_MULTIDEVICE=1, autoConnect=1 */ 0, 0), /* Default priority and stack size */ rmmAPI(addressIP, addressPort, configFile) { createEPICSParams(); //Creates the parameters with createParam()... updateRMMInfos(); // Updates some important parameters fetchRingStatus(); } (...) asynStatus RMM::writeInt32(asynUser *pasynUser, epicsInt32 value)
(...){ asynStatus status = asynSuccess; const char *functionName = "writeInt32"; const char *paramName; int addr; int function = pasynUser->reason; getAddress(pasynUser, &addr); getParamName(addr, function, ¶mName); if ((value) && (function == rmmResetPacketCounts_)) { rmmAPI.resetPacketCounters(); } Etc. From: Mark Rivers <rivers at cars.uchicago.edu>
Sent: 30 January 2025 23:23:51 To: tech-talk at aps.anl.gov; Marco Filho Subject: Re: Understanding ASYN_CANBLOCK
Hi Marco,
If you have not already done so you should read the documentation here:
You should always use ASYN_CANBLOCK for any driver which takes more than a few 10s of microseconds to complete an I/O request from device support. If you do not use ASYN_CANBLOCK then the thread that calls you driver will be blocked until the I/O operation
is complete. This includes the scan tasks (1 second, 2 second, etc.), channel access server threads, etc.
If you are seeing freezing when you use ASYN_CANBLOCK then either you are starved for CPU (what does "top" show?) or something is wrong. EPICS areaDetector drivers and plugins all use ASYN_CANBLOCK and they do thousands of callbacks to device support per second
with no problem with performance.
Can you describe the driver and the number of I/O operations per second in more detail?
One way to reduce CPU is to write a thread in your driver that reads all of the parameters. The records associated with those parameters have SCAN=I/O Intr. When your driver does callParamCallbacks() only the parameters that have changed will have their records
process. If most parameters only change infrequently that can drastically reduce the amount of record processing.
Mark
From: Tech-talk <tech-talk-bounces at aps.anl.gov> on behalf of Marco Filho via Tech-talk <tech-talk at aps.anl.gov>
Sent: Thursday, January 30, 2025 4:02 PM To: tech-talk at aps.anl.gov <tech-talk at aps.anl.gov> Subject: Understanding ASYN_CANBLOCK Hi all. I'm writing this e-mail to ask for help understanding ASYN_CANBLOCK. asynPortDriver(portName, 1,
asynInt8ArrayMask | asynInt32Mask | asynInt32ArrayMask | asynInt64Mask | asynFloat64Mask | asynOctetMask | asynDrvUserMask, // Interfaces that we implement asynInt8ArrayMask | asynInt32Mask | asynInt32ArrayMask | asynInt64Mask | asynFloat64Mask | asynOctetMask, // Interfaces that do callbacks ASYN_MULTIDEVICE | ASYN_CANBLOCK, 1, /* ASYN_CANBLOCK=1, ASYN_MULTIDEVICE=1, autoConnect=1 */ 0, 0), /* Default priority and stack size */ And removing the flag caused the OPI and shell to not freeze. By visually and manually inspecting I also have the feeling that the parameters are being properly scanned and that I'm not leaving anything behind. The question is: by removing ASYN_CANBLOCK am I actually distributing the readings in time or am I doing something else? Are the requests actually being executed in parallel? Thanks for any elucidations, Marco |